diff options
author | Michele Calgaro <[email protected]> | 2024-10-13 11:56:14 +0900 |
---|---|---|
committer | Michele Calgaro <[email protected]> | 2024-10-29 21:58:42 +0900 |
commit | 2879ff70be9271550477982a1a6371714db38562 (patch) | |
tree | c2054149dba923ab080fe7093432c7663a990111 /src | |
parent | 3eb38d2556f676d1027746f20bf12a1dd74451ef (diff) | |
download | krecipes-2879ff70be9271550477982a1a6371714db38562.tar.gz krecipes-2879ff70be9271550477982a1a6371714db38562.zip |
Rearrange folders structure to remove unnecessary 'krecipes' second level subfolder
Signed-off-by: Michele Calgaro <[email protected]>
(cherry picked from commit 0c8ed6c9a4000af8f48581a81c4b5c2f5b9fd502)
Diffstat (limited to 'src')
294 files changed, 50332 insertions, 0 deletions
diff --git a/src/Makefile.am b/src/Makefile.am new file mode 100644 index 0000000..61daaec --- /dev/null +++ b/src/Makefile.am @@ -0,0 +1,62 @@ +## Makefile.am for krecipes + +# this is the program that gets installed. it's name is used for all +# of the other Makefile.am variables +bin_PROGRAMS = krecipes + +SUBDIRS = backends importers widgets dialogs exporters datablocks tests + +# set the include path for X, tqt and TDE +INCLUDES = -I$(srcdir) -I$(srcdir)/backends -I$(srcdir)/backends/SQLite $(all_includes) + +# the library search path. +krecipes_LDFLAGS = $(KDE_RPATH) $(LIB_TDEPARTS) $(LIB_TDECORE) $(LIB_TDEUI) $(LIB_TDEIO) -ltdefx -lDCOP $(LIB_TQT) $(all_libraries) + + +# Check for optional libs +if link_lib_SQLITE +qsqlite_libadds = backends/SQLite/libkrecsqlite.la $(SQLITE_LIB) +endif + +if link_lib_MYSQL +mysql_libadds =backends/MySQL/libkrecmysql.la +endif + +if link_lib_POSTGRESQL +psql_libadds =backends/PostgreSQL/libkrecpsql.la +endif + +# the shared libraries to link against. +krecipes_LDADD = \ + backends/libkrecipesdbs.la exporters/libkrecipesexporters.la \ + importers/libkrecipesimporters.la widgets/libkrecipeswidgets.la dialogs/libkrecipesdialogs.la \ + widgets/libkrecipeswidgets.la datablocks/libdatablocks.la \ + $(qsqlite_libadds) $(mysql_libadds) $(psql_libadds) $(LIB_TDEHTML) $(LIB_TDESPELL) $(LIB_TDEPARTS) -ltdefx + +# which sources should be compiled for krecipes +krecipes_SOURCES = \ + main.cpp krecipes.cpp krecipesview.cpp pref.cpp \ + krecipesiface.skel krecipesdbiface.skel \ + propertycalculator.cpp setupwizard.cpp \ + shoppingcalculator.cpp kstartuplogo.cpp \ + recipeactionshandler.cpp \ + recipefilter.cpp \ + convert_sqlite3.cpp klomanager.cpp + +# let automoc handle all of the meta source files (moc) +METASOURCES = AUTO + +KDE_ICON = krecipes + +# this is where the kdelnk file will go +kdelnkdir = $(kde_appsdir)/Utilities +kdelnk_DATA = krecipes.desktop + +# this is where the XML-GUI resource file goes +rcdir = $(kde_datadir)/krecipes +rc_DATA = krecipesui.rc + +messages: rc.cpp + find ./ -name \*.ui | xargs --no-run-if-empty $(EXTRACTRC) >> rc.cpp + $(XGETTEXT) rc.cpp `find ./ -name \*.cpp` -o $(podir)/krecipes.pot + rm -f rc.cpp diff --git a/src/backends/Makefile.am b/src/backends/Makefile.am new file mode 100644 index 0000000..d9732a0 --- /dev/null +++ b/src/backends/Makefile.am @@ -0,0 +1,42 @@ +## Makefile.am for krecipes + +# this is the program that gets installed. it's name is used for all +# of the other Makefile.am variables + +# set the include path for X, tqt and TDE +INCLUDES = -I$(srcdir) -I$(srcdir)/.. $(all_includes) + +# Check for optional database libs + + +if link_lib_SQLITE +sqlite_subdirs=SQLite +sqlite_libs=SQLite/libkrecsqlite.la +endif + +if link_lib_MYSQL +mysql_subdirs=MySQL +mysql_libs=MySQL/libkrecmysql.la +endif + +if link_lib_POSTGRESQL +psql_subdirs=PostgreSQL +psql_libs=PostgreSQL/libkrecpsql.la +endif + +# Optional subdirectories + +SUBDIRS=$(sqlite_subdirs) $(mysql_subdirs) $(psql_subdirs) + +# Instructions for building the convenience library +noinst_LTLIBRARIES=libkrecipesdbs.la +libkrecipesdbs_la_SOURCES=recipedb.cpp qsqlrecipedb.cpp progressinterface.cpp +libkrecipesdbs_la_METASOURCES=AUTO + +libkrecipesdbs_la_LIBADD= $(mysql_libs) $(sqlite_libs) $(psql_libs) + +#the library search path. +libkrecipesdbs_la_LDFLAGS = $(KDE_RPATH) $(all_libraries) + +#install the following headers +# include_HEADERS = recipedb.h diff --git a/src/backends/MySQL/Makefile.am b/src/backends/MySQL/Makefile.am new file mode 100644 index 0000000..7134130 --- /dev/null +++ b/src/backends/MySQL/Makefile.am @@ -0,0 +1,18 @@ +## Makefile.am for krecipes + +# this is the program that gets installed. it's name is used for all +# of the other Makefile.am variables + +# set the include path for X, tqt and TDE +INCLUDES = -I$(srcdir) -I$(srcdir)/.. -I$(srcdir)/../.. $(all_includes) + + +# Instructions for building the convenience library +noinst_LTLIBRARIES=libkrecmysql.la +libkrecmysql_la_SOURCES=mysqlrecipedb.cpp +libkrecmysql_la_METASOURCES=AUTO + + +#the library search path. +libkrecmysql_la_LDFLAGS = $(KDE_RPATH) $(all_libraries) + diff --git a/src/backends/MySQL/mysqlrecipedb.cpp b/src/backends/MySQL/mysqlrecipedb.cpp new file mode 100644 index 0000000..7d26a3d --- /dev/null +++ b/src/backends/MySQL/mysqlrecipedb.cpp @@ -0,0 +1,554 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* Copyright (C) 2006 Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "mysqlrecipedb.h" + +#include <kdebug.h> +#include <kstandarddirs.h> +#include <tdetempfile.h> +#include <tdelocale.h> +#include <tdeconfig.h> +#include <tdeglobal.h> + +MySQLRecipeDB::MySQLRecipeDB( const TQString &host, const TQString &user, const TQString &pass, const TQString &DBname, int port ) : TQSqlRecipeDB( host, user, pass, DBname, port ) +{} + +MySQLRecipeDB::~MySQLRecipeDB() +{} + +void MySQLRecipeDB::createDB() +{ + TQString real_db_name = database->databaseName(); + + //we have to be connected to some database in order to create the Krecipes database + //so long as the permissions given are allowed access to "mysql', this works + database->setDatabaseName( "mysql" ); + if ( database->open() ) { + // Create the Database (Note: needs permissions) + //FIXME: I've noticed certain characters cause this to fail (such as '-'). Somehow let the user know. + TQSqlQuery query( TQString( "CREATE DATABASE %1" ).arg( real_db_name ), database ); + if ( !query.isActive() ) + kdDebug() << "create query failed: " << database->lastError().databaseText() << endl; + + database->close(); + } + else + kdDebug() << "create open failed: " << database->lastError().databaseText() << endl; + + database->setDatabaseName( real_db_name ); +} + +TQStringList MySQLRecipeDB::backupCommand() const +{ + TDEConfig *config = TDEGlobal::config(); + config->setGroup("Server"); + + TQStringList command; + command<<config->readEntry( "MySQLDumpPath", "mysqldump" )<<"-q"; + + TQString pass = config->readEntry("Password", TQString::null); + if ( !pass.isEmpty() ) + command<<"-p"+pass; + + TQString user = config->readEntry("Username", TQString::null); + command<<"-u"+user; + + command<<"-h"+config->readEntry("Host", "localhost"); + + int port = config->readNumEntry("Port", 0); + if ( port > 0 ) + command<<"-P"+TQString::number(port); + + command<<database->databaseName(); + return command; +} + +TQStringList MySQLRecipeDB::restoreCommand() const +{ + TDEConfig *config = TDEGlobal::config(); + config->setGroup("Server"); + + TQStringList command; + command<<config->readEntry( "MySQLPath", "mysql" ); + + TQString pass = config->readEntry("Password", TQString::null); + if ( !pass.isEmpty() ) + command<<"-p"+pass; + + TQString user = config->readEntry("Username", TQString::null); + command<<"-u"+user; + + int port = config->readNumEntry("Port", 0); + if ( port > 0 ) + command<<"-P"+TQString::number(port); + + command<<"-h"+config->readEntry("Host", "localhost"); + + command<<database->databaseName(); + return command; +} + +void MySQLRecipeDB::createTable( const TQString &tableName ) +{ + + TQStringList commands; + + if ( tableName == "recipes" ) + commands << TQString( "CREATE TABLE recipes (id INTEGER NOT NULL AUTO_INCREMENT,title VARCHAR(%1), yield_amount FLOAT, yield_amount_offset FLOAT, yield_type_id int(11) DEFAULT '-1', instructions TEXT, photo BLOB, prep_time TIME, ctime TIMESTAMP, mtime TIMESTAMP, atime TIMESTAMP, PRIMARY KEY (id));" ).arg( maxRecipeTitleLength() ); + + else if ( tableName == "ingredients" ) + commands << TQString( "CREATE TABLE ingredients (id INTEGER NOT NULL AUTO_INCREMENT, name VARCHAR(%1), PRIMARY KEY (id));" ).arg( maxIngredientNameLength() ); + + else if ( tableName == "ingredient_list" ) + commands << "CREATE TABLE ingredient_list (id INTEGER NOT NULL AUTO_INCREMENT, recipe_id INTEGER, ingredient_id INTEGER, amount FLOAT, amount_offset FLOAT, unit_id INTEGER, order_index INTEGER, group_id INTEGER, substitute_for INTEGER, PRIMARY KEY(id), INDEX ridil_index(recipe_id), INDEX iidil_index(ingredient_id), INDEX gidil_index(group_id))"; + + else if ( tableName == "unit_list" ) + commands << "CREATE TABLE unit_list (ingredient_id INTEGER, unit_id INTEGER);"; + + else if ( tableName == "units" ) + commands << TQString( "CREATE TABLE units (id INTEGER NOT NULL AUTO_INCREMENT, name VARCHAR(%1), name_abbrev VARCHAR(%2), plural VARCHAR(%3), plural_abbrev VARCHAR(%4), type INTEGER NOT NULL DEFAULT 0, PRIMARY KEY (id));" ) + .arg( maxUnitNameLength() ).arg( maxUnitNameLength() ).arg( maxUnitNameLength() ).arg( maxUnitNameLength() ); + + else if ( tableName == "prep_methods" ) + commands << TQString( "CREATE TABLE prep_methods (id INTEGER NOT NULL AUTO_INCREMENT, name VARCHAR(%1), PRIMARY KEY (id));" ).arg( maxPrepMethodNameLength() ); + + else if ( tableName == "prep_method_list" ) + commands << "CREATE TABLE prep_method_list (ingredient_list_id int(11) NOT NULL,prep_method_id int(11) NOT NULL, order_index int(11), INDEX iid_index (ingredient_list_id), INDEX pid_index (prep_method_id));"; + + else if ( tableName == "ingredient_info" ) + commands << "CREATE TABLE ingredient_info (ingredient_id INTEGER, property_id INTEGER, amount FLOAT, per_units INTEGER);"; + + else if ( tableName == "ingredient_properties" ) + commands << "CREATE TABLE ingredient_properties (id INTEGER NOT NULL AUTO_INCREMENT,name VARCHAR(20), units VARCHAR(20), PRIMARY KEY (id));"; + + else if ( tableName == "ingredient_weights" ) + commands << "CREATE TABLE ingredient_weights (id INTEGER NOT NULL AUTO_INCREMENT, ingredient_id INTEGER NOT NULL, amount FLOAT, unit_id INTEGER, weight FLOAT, weight_unit_id INTEGER, prep_method_id INTEGER, PRIMARY KEY (id), INDEX(ingredient_id), INDEX(unit_id), INDEX(weight_unit_id), INDEX(prep_method_id) );"; + + else if ( tableName == "units_conversion" ) + commands << "CREATE TABLE units_conversion (unit1_id INTEGER, unit2_id INTEGER, ratio FLOAT);"; + + else if ( tableName == "categories" ) + commands << TQString( "CREATE TABLE categories (id int(11) NOT NULL auto_increment, name varchar(%1) default NULL, parent_id int(11) NOT NULL default -1, PRIMARY KEY (id), INDEX parent_id_index(parent_id));" ).arg( maxCategoryNameLength() ); + + else if ( tableName == "category_list" ) + commands << "CREATE TABLE category_list (recipe_id int(11) NOT NULL,category_id int(11) NOT NULL, INDEX rid_index (recipe_id), INDEX cid_index (category_id));"; + + else if ( tableName == "authors" ) + commands << TQString( "CREATE TABLE authors (id int(11) NOT NULL auto_increment, name varchar(%1) default NULL,PRIMARY KEY (id));" ).arg( maxAuthorNameLength() ); + + else if ( tableName == "author_list" ) + commands << "CREATE TABLE author_list (recipe_id int(11) NOT NULL,author_id int(11) NOT NULL);"; + + else if ( tableName == "db_info" ) { + commands << "CREATE TABLE db_info (ver FLOAT NOT NULL,generated_by varchar(200) default NULL);"; + commands << TQString( "INSERT INTO db_info VALUES(%1,'Krecipes %2');" ).arg( latestDBVersion() ).arg( krecipes_version() ); + } + else if ( tableName == "ingredient_groups" ) { + commands << TQString( "CREATE TABLE `ingredient_groups` (`id` int(11) NOT NULL auto_increment, `name` varchar(%1), PRIMARY KEY (`id`));" ).arg( maxIngGroupNameLength() ); + } + else if ( tableName == "yield_types" ) { + commands << TQString( "CREATE TABLE `yield_types` (`id` int(11) NOT NULL auto_increment, `name` varchar(%1), PRIMARY KEY (`id`));" ).arg( 20 ); + } + + else if ( tableName == "ratings" ) + commands << "CREATE TABLE ratings (id INTEGER NOT NULL AUTO_INCREMENT, recipe_id int(11) NOT NULL, comment TEXT, rater TEXT, created TIMESTAMP, PRIMARY KEY (id));"; + + else if ( tableName == "rating_criteria" ) + commands << "CREATE TABLE rating_criteria (id INTEGER NOT NULL AUTO_INCREMENT, name TEXT, PRIMARY KEY (id));"; + + else if ( tableName == "rating_criterion_list" ) + commands << "CREATE TABLE rating_criterion_list (rating_id INTEGER NOT NULL, rating_criterion_id INTEGER, stars FLOAT);"; + + else + return ; + + TQSqlQuery databaseToCreate( TQString::null, database ); + + // execute the queries + for ( TQStringList::const_iterator it = commands.begin(); it != commands.end(); ++it ) + databaseToCreate.exec( ( *it ) ); +} + +void MySQLRecipeDB::portOldDatabases( float version ) +{ + kdDebug() << "Current database version is..." << version << "\n"; + TQString command; + + // Note that version no. means the version in which this DB structure + // was introduced. To work with SVN users, the database will be incrementally + // upgraded for each change made between releases (e.g. 0.81, 0.82,... are + // what will become 0.9) + + if ( tqRound(version*10) < 3 ) // The database was generated with a version older than v 0.3. First update to 0.3 version + { + + // Add new columns to existing tables (creating new tables is not necessary. Integrity check does that before) + command = "ALTER TABLE recipes ADD COLUMN persons int(11) AFTER title;"; + TQSqlQuery tableToAlter( command, database ); + + // Set the version to the new one (0.3) + + command = "DELETE FROM db_info;"; // Remove previous version records if they exist + tableToAlter.exec( command ); + command = "INSERT INTO db_info VALUES(0.3,'Krecipes 0.4');"; // Set the new version + tableToAlter.exec( command ); + } + + if ( tqRound(version*10) < 4 ) // Upgrade to the current DB version 0.4 + { + + // Add new columns to existing tables (creating any new tables is not necessary. Integrity check does that before) + command = "ALTER TABLE ingredient_list ADD COLUMN order_index int(11) AFTER unit_id;"; + TQSqlQuery tableToAlter( command, database ); + + // Missing indexes in the previous versions + command = "CREATE index rid_index ON category_list(recipe_id)"; + tableToAlter.exec( command ); + + command = "CREATE index cid_index ON category_list(category_id)"; + tableToAlter.exec( command ); + + command = "CREATE index ridil_index ON ingredient_list(recipe_id)"; + tableToAlter.exec( command ); + + command = "CREATE index iidil_index ON ingredient_list(ingredient_id)"; + tableToAlter.exec( command ); + + // Port data + + //*1:: Recipes have always category -1 to speed up searches (no JOINs needed) + command = "SELECT r.id FROM recipes r;"; // Find all recipes + TQSqlQuery categoryToAdd( TQString::null, database ); + tableToAlter.exec( command ); + if ( tableToAlter.isActive() ) + { + while ( tableToAlter.next() ) { + int recipeId = tableToAlter.value( 0 ).toInt(); + TQString cCommand = TQString( "INSERT INTO category_list VALUES (%1,-1);" ).arg( recipeId ); + categoryToAdd.exec( cCommand ); + + emit progress(); + } + } + + // Set the version to the new one (0.4) + + command = "DELETE FROM db_info;"; // Remove previous version records if they exist + tableToAlter.exec( command ); + command = "INSERT INTO db_info VALUES(0.4,'Krecipes 0.4');"; // Set the new version + tableToAlter.exec( command ); + } + + if ( tqRound(version*10) < 5 ) { + command = TQString( "CREATE TABLE prep_methods (id INTEGER NOT NULL AUTO_INCREMENT, name VARCHAR(%1), PRIMARY KEY (id));" ).arg( maxPrepMethodNameLength() ); + TQSqlQuery tableToAlter( command, database ); + + command = "ALTER TABLE ingredient_list ADD COLUMN prep_method_id int(11) AFTER unit_id;"; + tableToAlter.exec( command ); + command = "UPDATE ingredient_list SET prep_method_id=-1 WHERE prep_method_id IS NULL;"; + tableToAlter.exec( command ); + + command = "ALTER TABLE authors MODIFY name VARCHAR(50);"; + tableToAlter.exec( command ); + command = "ALTER TABLE categories MODIFY name VARCHAR(40);"; + tableToAlter.exec( command ); + + // Set the version to the new one (0.5) + command = "DELETE FROM db_info;"; // Remove previous version records if they exist + tableToAlter.exec( command ); + command = "INSERT INTO db_info VALUES(0.5,'Krecipes 0.5');"; + tableToAlter.exec( command ); + } + + if ( tqRound(version*10) < 6 ) { + command = "ALTER TABLE categories ADD COLUMN parent_id int(11) NOT NULL default '-1' AFTER name;"; + TQSqlQuery tableToAlter( command, database ); + + command = "DELETE FROM db_info;"; // Remove previous version records if they exist + tableToAlter.exec( command ); + command = "INSERT INTO db_info VALUES(0.6,'Krecipes 0.6');"; + tableToAlter.exec( command ); + } + + if ( tqRound(version*100) < 61 ) { + TQString command = "ALTER TABLE `recipes` ADD COLUMN `prep_time` TIME DEFAULT NULL"; + TQSqlQuery tableToAlter( command, database ); + + command = "DELETE FROM db_info;"; // Remove previous version records if they exist + tableToAlter.exec( command ); + command = "INSERT INTO db_info VALUES(0.61,'Krecipes 0.6');"; + tableToAlter.exec( command ); + } + + if ( tqRound(version*100) < 62 ) { + TQString command = "ALTER TABLE `ingredient_list` ADD COLUMN `group_id` int(11) default '-1' AFTER order_index;"; + TQSqlQuery tableToAlter( command, database ); + + command = "DELETE FROM db_info;"; // Remove previous version records if they exist + tableToAlter.exec( command ); + command = "INSERT INTO db_info VALUES(0.62,'Krecipes 0.7');"; + tableToAlter.exec( command ); + } + + if ( tqRound(version*100) < 63 ) { + TQString command = "ALTER TABLE `units` ADD COLUMN `plural` varchar(20) DEFAULT NULL AFTER name;"; + TQSqlQuery tableToAlter( command, database ); + + TQSqlQuery result( "SELECT id,name FROM units WHERE plural IS NULL", database ); + if ( result.isActive() ) { + while ( result.next() ) { + command = "UPDATE units SET plural='" + result.value( 1 ).toString() + "' WHERE id=" + TQString::number( result.value( 0 ).toInt() ); + TQSqlQuery query( command, database ); + + emit progress(); + } + } + + command = "DELETE FROM db_info;"; // Remove previous version records if they exist + tableToAlter.exec( command ); + command = "INSERT INTO db_info VALUES(0.63,'Krecipes 0.7');"; + tableToAlter.exec( command ); + } + + if ( tqRound(version*10) < 7 ) { //simply call 0.63 -> 0.7 + TQString command = "UPDATE db_info SET ver='0.7';"; + TQSqlQuery query( command, database ); + } + + if ( tqRound(version*100) < 81 ) { + TQString command = "ALTER TABLE `ingredient_list` ADD COLUMN `amount_offset` FLOAT DEFAULT '0' AFTER amount;"; + TQSqlQuery tableToAlter( command, database ); + + command = "UPDATE db_info SET ver='0.81',generated_by='Krecipes SVN (20050816)';"; + tableToAlter.exec( command ); + } + + if ( tqRound(version*100) < 82 ) { + TQString command = "ALTER TABLE `recipes` ADD COLUMN `yield_amount` FLOAT DEFAULT '0' AFTER persons;"; + TQSqlQuery tableToAlter( command, database ); + + command = "ALTER TABLE `recipes` ADD COLUMN `yield_amount_offset` FLOAT DEFAULT '0' AFTER yield_amount;"; + tableToAlter.exec(command); + + command = "ALTER TABLE `recipes` ADD COLUMN `yield_type_id` INTEGER DEFAULT '-1' AFTER yield_amount_offset;"; + tableToAlter.exec(command); + + TQSqlQuery result( "SELECT id,persons FROM recipes", database ); + if ( result.isActive() ) { + while ( result.next() ) { + command = "UPDATE recipes SET yield_amount='" + TQString::number( result.value( 1 ).toInt() ) + "' WHERE id=" + TQString::number( result.value( 0 ).toInt() ); + TQSqlQuery query( command, database ); + + emit progress(); + } + } + + command = "ALTER TABLE `recipes` DROP COLUMN `persons`;"; + tableToAlter.exec( command ); + + command = "UPDATE db_info SET ver='0.82',generated_by='Krecipes SVN (20050902)';"; + tableToAlter.exec( command ); + } + + if ( tqRound(version*100) < 83 ) { + database->transaction(); + + //====add a id columns to 'ingredient_list' to identify it for the prep method list + database->exec( "RENAME TABLE ingredient_list TO ingredient_list_copy;" ); + database->exec( "CREATE TABLE ingredient_list (id INTEGER NOT NULL AUTO_INCREMENT, recipe_id INTEGER, ingredient_id INTEGER, amount FLOAT, amount_offset FLOAT, unit_id INTEGER, order_index INTEGER, group_id INTEGER, PRIMARY KEY(id), INDEX ridil_index(recipe_id), INDEX iidil_index(ingredient_id));" ); + + TQSqlQuery copyQuery = database->exec( "SELECT recipe_id,ingredient_id,amount,amount_offset,unit_id,prep_method_id,order_index,group_id FROM ingredient_list_copy" ); + if ( copyQuery.isActive() ) { + while ( copyQuery.next() ) { + TQSqlQuery query(TQString::null,database); + query.prepare( "INSERT INTO ingredient_list VALUES (NULL, ?, ?, ?, ?, ?, ?, ?)" ); + query.addBindValue( copyQuery.value( 0 ) ); + query.addBindValue( copyQuery.value( 1 ) ); + query.addBindValue( copyQuery.value( 2 ) ); + query.addBindValue( copyQuery.value( 3 ) ); + query.addBindValue( copyQuery.value( 4 ) ); + query.addBindValue( copyQuery.value( 6 ) ); + query.addBindValue( copyQuery.value( 7 ) ); + query.exec(); + + int prep_method_id = copyQuery.value( 5 ).toInt(); + if ( prep_method_id != -1 ) { + query.prepare( "INSERT INTO prep_method_list VALUES (?, ?, ?);" ); + query.addBindValue( lastInsertID() ); + query.addBindValue( prep_method_id ); + query.addBindValue( 1 ); + query.exec(); + } + + emit progress(); + } + } + database->exec( "DROP TABLE ingredient_list_copy" ); + + database->exec( "UPDATE db_info SET ver='0.83',generated_by='Krecipes SVN (20050909)';" ); + + if ( !database->commit() ) + kdDebug()<<"Update to 0.83 failed. Maybe you should try again."<<endl; + } + + if ( tqRound(version*100) < 84 ) { + database->transaction(); + + database->exec( "ALTER TABLE recipes ADD COLUMN ctime TIMESTAMP;" ); + database->exec( "ALTER TABLE recipes ADD COLUMN mtime TIMESTAMP;" ); + database->exec( "ALTER TABLE recipes ADD COLUMN atime TIMESTAMP;" ); + + database->exec( "UPDATE recipes SET ctime=CURRENT_TIMESTAMP, mtime=CURRENT_TIMESTAMP, atime=CURRENT_TIMESTAMP;" ); + + database->exec( "UPDATE db_info SET ver='0.84',generated_by='Krecipes SVN (20050913)';" ); + + if ( !database->commit() ) + kdDebug()<<"Update to 0.84 failed. Maybe you should try again."<<endl; + } + + if ( tqRound(version*100) < 85 ) { + database->transaction(); + + TQSqlQuery query( "SELECT id,photo FROM recipes", database ); + + if ( query.isActive() ) { + while ( query.next() ) { + storePhoto( query.value(0).toInt(), query.value(1).toByteArray() ); + + emit progress(); + } + } + + + database->exec( "UPDATE db_info SET ver='0.85',generated_by='Krecipes SVN (20050926)';" ); + if ( !database->commit() ) + kdDebug()<<"Update to 0.85 failed. Maybe you should try again."<<endl; + } + + if ( tqRound(version*100) < 86 ) { + database->transaction(); + + database->exec( "ALTER TABLE ingredient_list ADD INDEX (group_id)" ); + + TQSqlQuery query( "SELECT id,name FROM ingredient_groups ORDER BY name", database ); + + TQString last; + int lastID; + if ( query.isActive() ) { + while ( query.next() ) { + TQString name = query.value(1).toString(); + int id = query.value(0).toInt(); + if ( last == name ) { + TQString command = TQString("UPDATE ingredient_list SET group_id=%1 WHERE group_id=%2").arg(lastID).arg(id); + database->exec(command); + + command = TQString("DELETE FROM ingredient_groups WHERE id=%1").arg(id); + database->exec(command); + } + last = name; + lastID = id; + + emit progress(); + } + } + + database->exec( "UPDATE db_info SET ver='0.86',generated_by='Krecipes SVN (20050928)';" ); + if ( !database->commit() ) + kdDebug()<<"Update to 0.86 failed. Maybe you should try again."<<endl; + } + + if ( tqRound(version*100) < 87 ) { + //Load this default data so the user knows what rating criteria is + database->exec( TQString("INSERT INTO rating_criteria VALUES (1,'%1')").arg(i18n("Overall")) ); + database->exec( TQString("INSERT INTO rating_criteria VALUES (2,'%1')").arg(i18n("Taste") ) ); + database->exec( TQString("INSERT INTO rating_criteria VALUES (3,'%1')").arg(i18n("Appearance") ) ); + database->exec( TQString("INSERT INTO rating_criteria VALUES (4,'%1')").arg(i18n("Originality") ) ); + database->exec( TQString("INSERT INTO rating_criteria VALUES (5,'%1')").arg(i18n("Ease of Preparation") ) ); + + database->exec( "UPDATE db_info SET ver='0.87',generated_by='Krecipes SVN (20051014)'" ); + } + + if ( tqRound(version*100) < 90 ) { + database->exec("UPDATE db_info SET ver='0.9',generated_by='Krecipes 0.9'"); + } + + if ( tqRound(version*100) < 91 ) { + database->exec("CREATE index parent_id_index ON categories(parent_id)"); + database->exec("UPDATE db_info SET ver='0.91',generated_by='Krecipes SVN (20060526)'"); + } + + if ( tqRound(version*100) < 92 ) { + database->transaction(); + + database->exec( "ALTER TABLE units ADD COLUMN name_abbrev VARCHAR(20) AFTER name"); + database->exec( "ALTER TABLE units ADD COLUMN plural_abbrev VARCHAR(20) AFTER plural"); + + database->exec("UPDATE db_info SET ver='0.92',generated_by='Krecipes SVN (20060609)'"); + if ( !database->commit() ) + kdDebug()<<"Update to 0.92 failed. Maybe you should try again."<<endl; + } + + if ( tqRound(version*100) < 93 ) { + database->transaction(); + + database->exec( "ALTER TABLE ingredient_list ADD COLUMN substitute_for INTEGER AFTER group_id"); + + database->exec("UPDATE db_info SET ver='0.93',generated_by='Krecipes SVN (20060615)'"); + if ( !database->commit() ) + kdDebug()<<"Update to 0.93 failed. Maybe you should try again."<<endl; + } + + if ( tqRound(version*100) < 94 ) { + database->transaction(); + + database->exec( "ALTER TABLE units ADD COLUMN type INTEGER NOT NULL DEFAULT 0 AFTER plural_abbrev"); + + database->exec("UPDATE db_info SET ver='0.94',generated_by='Krecipes SVN (20060712)'"); + if ( !database->commit() ) + kdDebug()<<"Update to 0.94 failed. Maybe you should try again."<<endl; + } + + if ( tqRound(version*100) < 95 ) { + database->exec( "DROP TABLE ingredient_weights" ); + createTable( "ingredient_weights" ); + database->exec( "UPDATE db_info SET ver='0.95',generated_by='Krecipes SVN (20060726)'" ); + } +} + +int MySQLRecipeDB::lastInsertID() +{ + TQSqlQuery lastInsertID( "SELECT LAST_INSERT_ID();", database ); + + int id = -1; + if ( lastInsertID.isActive() && lastInsertID.next() ) + id = lastInsertID.value( 0 ).toInt(); + + return id; +} + +void MySQLRecipeDB::givePermissions( const TQString &dbName, const TQString &username, const TQString &password, const TQString &clientHost ) +{ + TQString command; + + if ( !password.isEmpty() ) + command = TQString( "GRANT ALL ON %1.* TO '%2'@'%3' IDENTIFIED BY '%4';" ).arg( dbName ).arg( username ).arg( clientHost ).arg( password ); + else + command = TQString( "GRANT ALL ON %1.* TO '%2'@'%3';" ).arg( dbName ).arg( username ).arg( clientHost ); + + kdDebug() << "I'm doing the query to setup permissions\n"; + + TQSqlQuery permissionsToSet( command, database ); +} + +#include "mysqlrecipedb.moc" diff --git a/src/backends/MySQL/mysqlrecipedb.h b/src/backends/MySQL/mysqlrecipedb.h new file mode 100644 index 0000000..ee93242 --- /dev/null +++ b/src/backends/MySQL/mysqlrecipedb.h @@ -0,0 +1,56 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + + +#ifndef MYSQLRECIPEDB_H +#define MYSQLRECIPEDB_H + +#define MYSQL_DRIVER "TQMYSQL3" + +#include "qsqlrecipedb.h" + +/** +@author Unai Garro +*/ +class MySQLRecipeDB : public TQSqlRecipeDB +{ + + TQ_OBJECT + +private: + void createDB( void ); + +public: + MySQLRecipeDB( const TQString &host, const TQString &user = TQString::null, const TQString &pass = TQString::null, const TQString &DBName = DEFAULT_DB_NAME, int port = 0 ); + ~MySQLRecipeDB( void ); + + int lastInsertID(); + + void createTable( const TQString &tableName ); + void givePermissions( const TQString &dbName, const TQString &username, const TQString &password = TQString::null, const TQString &clientHost = "localhost" ); + +protected: + TQString qsqlDriverPlugin() const + { + return MYSQL_DRIVER; + } + +private: + void portOldDatabases( float version ); + TQStringList backupCommand() const; + TQStringList restoreCommand() const; +}; + + + + +#endif diff --git a/src/backends/PostgreSQL/Makefile.am b/src/backends/PostgreSQL/Makefile.am new file mode 100644 index 0000000..dac2623 --- /dev/null +++ b/src/backends/PostgreSQL/Makefile.am @@ -0,0 +1,17 @@ +## Makefile.am for krecipes + +# this is the program that gets installed. it's name is used for all +# of the other Makefile.am variables + +# set the include path for X, tqt and TDE +INCLUDES = -I$(srcdir) -I$(srcdir)/.. -I$(srcdir)/../.. $(all_includes) + + +# Instructions for building the convenience library +noinst_LTLIBRARIES=libkrecpsql.la +libkrecpsql_la_SOURCES=psqlrecipedb.cpp +libkrecpsql_la_METASOURCES=AUTO + + +#the library search path. +libkrecpsql_la_LDFLAGS = $(KDE_RPATH) $(all_libraries) diff --git a/src/backends/PostgreSQL/psqlrecipedb.cpp b/src/backends/PostgreSQL/psqlrecipedb.cpp new file mode 100644 index 0000000..c5e2129 --- /dev/null +++ b/src/backends/PostgreSQL/psqlrecipedb.cpp @@ -0,0 +1,560 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* Copyright (C) 2006 Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "psqlrecipedb.h" + +#include <kdebug.h> +#include <kstandarddirs.h> +#include <tdetempfile.h> +#include <tdelocale.h> +#include <tdeconfig.h> + +#include <tqvariant.h> + +//Note: PostgreSQL's database names are always lowercase +PSqlRecipeDB::PSqlRecipeDB( const TQString& host, const TQString& user, const TQString& pass, const TQString& DBname, int port ) : TQSqlRecipeDB( host, user, pass, DBname.lower(), port ) +{} + +PSqlRecipeDB::~PSqlRecipeDB() +{} + +void PSqlRecipeDB::createDB() +{ + TQString real_db_name = database->databaseName(); + + //we have to be connected to some database in order to create the Krecipes database + //so long as the permissions given are allowed access to "template1', this works + database->setDatabaseName( "template1" ); + if ( database->open() ) { + TQSqlQuery query( TQString( "CREATE DATABASE %1" ).arg( real_db_name ), database ); + if ( !query.isActive() ) + kdDebug() << "create query failed: " << database->lastError().databaseText() << endl; + + database->close(); + } + else + kdDebug() << "create open failed: " << database->lastError().databaseText() << endl; + + database->setDatabaseName( real_db_name ); +} + +TQStringList PSqlRecipeDB::backupCommand() const +{ + TDEConfig *config = TDEGlobal::config(); + config->setGroup("Server"); + + TQStringList command; + command<<config->readEntry( "PgDumpPath", "pg_dump" )<<"-d"<<database->databaseName() + <<"-U"<<config->readEntry( "Username" ); + + int port = config->readNumEntry( "Port", 0 ); + if ( port > 0 ) + command<<"-p"<<TQString::number(port); + + return command; +} + +TQStringList PSqlRecipeDB::restoreCommand() const +{ + TDEConfig *config = TDEGlobal::config(); + config->setGroup("Server"); + + TQStringList command; + command<<config->readEntry( "PsqlPath", "psql" )<<database->databaseName() + <<"-U"<<config->readEntry( "Username" ); + + int port = config->readNumEntry( "Port", 0 ); + if ( port > 0 ) + command<<"-p"<<TQString::number(port); + + return command; +} + +void PSqlRecipeDB::createTable( const TQString &tableName ) +{ + + TQStringList commands; + + if ( tableName == "recipes" ) + commands << "CREATE TABLE recipes (id SERIAL NOT NULL PRIMARY KEY,title CHARACTER VARYING, yield_amount FLOAT, yield_amount_offset FLOAT, yield_type_id INTEGER DEFAULT '-1', instructions TEXT, photo TEXT, prep_time TIME, ctime TIMESTAMP, mtime TIMESTAMP, atime TIMESTAMP );"; + + else if ( tableName == "ingredients" ) + commands << "CREATE TABLE ingredients (id SERIAL NOT NULL PRIMARY KEY, name CHARACTER VARYING);"; + + else if ( tableName == "ingredient_list" ) { + commands << "CREATE TABLE ingredient_list (id SERIAL NOT NULL PRIMARY KEY, recipe_id INTEGER, ingredient_id INTEGER, amount FLOAT, amount_offset FLOAT, unit_id INTEGER, order_index INTEGER, group_id INTEGER, substitute_for INTEGER);"; + commands << "CREATE INDEX ridil_index ON ingredient_list USING BTREE (recipe_id);"; + commands << "CREATE INDEX iidil_index ON ingredient_list USING BTREE (ingredient_id);"; + commands << "CREATE INDEX gidil_index ON ingredient_list USING BTREE (group_id);"; + } + + else if ( tableName == "unit_list" ) + commands << "CREATE TABLE unit_list (ingredient_id INTEGER, unit_id INTEGER);"; + + else if ( tableName == "units" ) + commands << "CREATE TABLE units (id SERIAL NOT NULL PRIMARY KEY, name CHARACTER VARYING, name_abbrev CHARACTER VARYING, plural CHARACTER VARYING, plural_abbrev CHARACTER VARYING, type INTEGER NOT NULL DEFAULT '0' );"; + + else if ( tableName == "prep_methods" ) + commands << "CREATE TABLE prep_methods (id SERIAL NOT NULL PRIMARY KEY, name CHARACTER VARYING);"; + else if ( tableName == "prep_method_list" ) { + commands << "CREATE TABLE prep_method_list (ingredient_list_id INTEGER NOT NULL,prep_method_id INTEGER NOT NULL, order_index INTEGER);"; + commands << "CREATE INDEX iid_index ON prep_method_list USING BTREE (ingredient_list_id);"; + commands << "CREATE INDEX pid_index ON prep_method_list USING BTREE (prep_method_id);"; + } + else if ( tableName == "ingredient_info" ) + commands << "CREATE TABLE ingredient_info (ingredient_id INTEGER, property_id INTEGER, amount FLOAT, per_units INTEGER);"; + + else if ( tableName == "ingredient_properties" ) + commands << "CREATE TABLE ingredient_properties (id SERIAL NOT NULL,name CHARACTER VARYING, units CHARACTER VARYING);"; + + else if ( tableName == "ingredient_weights" ) { + commands << "CREATE TABLE ingredient_weights (id SERIAL NOT NULL PRIMARY KEY, ingredient_id INTEGER NOT NULL, amount FLOAT, unit_id INTEGER, weight FLOAT, weight_unit_id INTEGER, prep_method_id INTEGER );" + << "CREATE INDEX weight_wid_index ON ingredient_weights USING BTREE (weight_unit_id)" + << "CREATE INDEX weight_pid_index ON ingredient_weights USING BTREE (prep_method_id)" + << "CREATE INDEX weight_uid_index ON ingredient_weights USING BTREE (unit_id)" + << "CREATE INDEX weight_iid_index ON ingredient_weights USING BTREE (ingredient_id)"; + } + + else if ( tableName == "units_conversion" ) + commands << "CREATE TABLE units_conversion (unit1_id INTEGER, unit2_id INTEGER, ratio FLOAT);"; + + else if ( tableName == "categories" ) { + commands << "CREATE TABLE categories (id SERIAL NOT NULL PRIMARY KEY, name CHARACTER VARYING default NULL, parent_id INTEGER NOT NULL default -1);"; + commands << "CREATE index parent_id_index ON categories USING BTREE(parent_id);"; + } + else if ( tableName == "category_list" ) { + commands << "CREATE TABLE category_list (recipe_id INTEGER NOT NULL,category_id INTEGER NOT NULL);"; + commands << "CREATE INDEX rid_index ON category_list USING BTREE (recipe_id);"; + commands << "CREATE INDEX cid_index ON category_list USING BTREE (category_id);"; + } + + else if ( tableName == "authors" ) + commands << "CREATE TABLE authors (id SERIAL NOT NULL PRIMARY KEY, name CHARACTER VARYING default NULL);"; + + else if ( tableName == "author_list" ) + commands << "CREATE TABLE author_list (recipe_id INTEGER NOT NULL,author_id INTEGER NOT NULL);"; + + else if ( tableName == "db_info" ) { + commands << "CREATE TABLE db_info (ver FLOAT NOT NULL,generated_by CHARACTER VARYING default NULL);"; + commands << TQString( "INSERT INTO db_info VALUES(%1,'Krecipes %2');" ).arg( latestDBVersion() ).arg( krecipes_version() ); + } + else if ( tableName == "ingredient_groups" ) { + commands << "CREATE TABLE ingredient_groups (id SERIAL NOT NULL PRIMARY KEY, name CHARACTER VARYING);"; + } + else if ( tableName == "yield_types" ) { + commands << "CREATE TABLE yield_types (id SERIAL NOT NULL PRIMARY KEY, name CHARACTER VARYING);"; + } + + else if ( tableName == "ratings" ) + commands << "CREATE TABLE ratings (id SERIAL NOT NULL PRIMARY KEY, recipe_id INTEGER NOT NULL, comment CHARACTER VARYING, rater CHARACTER VARYING, created TIMESTAMP);"; + + else if ( tableName == "rating_criteria" ) + commands << "CREATE TABLE rating_criteria (id SERIAL NOT NULL PRIMARY KEY, name CHARACTER VARYING);"; + + else if ( tableName == "rating_criterion_list" ) + commands << "CREATE TABLE rating_criterion_list (rating_id INTEGER NOT NULL, rating_criterion_id INTEGER, stars FLOAT);"; + else + return ; + + TQSqlQuery databaseToCreate( TQString::null, database ); + + // execute the queries + for ( TQStringList::const_iterator it = commands.begin(); it != commands.end(); ++it ) + databaseToCreate.exec( *it ); +} + +void PSqlRecipeDB::initializeData() +{ + TQSqlRecipeDB::initializeData(); + + TQSqlQuery updateSeq( "SELECT setval('units_id_seq',(SELECT COUNT(1) FROM units))", database ); + updateSeq.exec( "SELECT setval('categories_id_seq',(SELECT COUNT(1) FROM categories))" ); +} + +void PSqlRecipeDB::portOldDatabases( float version ) +{ + kdDebug() << "Current database version is..." << version << "\n"; + TQString command; + + if ( tqRound(version*10) < 7 ) { + //version added + } + + if ( tqRound(version*100) < 81 ) { + database->transaction(); + + addColumn("CREATE TABLE %1 (recipe_id INTEGER, ingredient_id INTEGER, amount FLOAT, %2 unit_id INTEGER, prep_method_id INTEGER, order_index INTEGER, group_id INTEGER);","amount_offset FLOAT","'0'","ingredient_list",3); + + TQSqlQuery query(TQString::null,database); + query.exec( "CREATE INDEX ridil_index ON ingredient_list USING BTREE (recipe_id);" ); + query.exec( "CREATE INDEX iidil_index ON ingredient_list USING BTREE (ingredient_id);"); + + query.exec( "UPDATE db_info SET ver='0.81',generated_by='Krecipes SVN (20050816)';" ); + + if ( !database->commit() ) + kdDebug()<<"Update to 0.81 failed. Maybe you should try again."<<endl; + } + + if ( tqRound(version*100) < 82 ) { + database->transaction(); + + //==================add a columns to 'recipes' to allow yield range + yield type + database->exec( "CREATE TABLE recipes_copy (id SERIAL NOT NULL PRIMARY KEY,title CHARACTER VARYING, persons INTEGER, instructions TEXT, photo TEXT, prep_time TIME);" ); + TQSqlQuery copyQuery = database->exec( "SELECT id,title,persons,instructions,photo,prep_time FROM recipes;" ); + if ( copyQuery.isActive() ) { + while ( copyQuery.next() ) { + TQSqlQuery query(TQString::null,database); + query.prepare( "INSERT INTO recipes_copy VALUES (?, ?, ?, ?, ?, ?)" ); + query.addBindValue( copyQuery.value( 0 ) ); + query.addBindValue( copyQuery.value( 1 ) ); + query.addBindValue( copyQuery.value( 2 ) ); + query.addBindValue( copyQuery.value( 3 ) ); + query.addBindValue( copyQuery.value( 4 ) ); + query.addBindValue( copyQuery.value( 5 ) ); + query.exec(); + + emit progress(); + } + } + database->exec( "DROP TABLE recipes" ); + database->exec( "CREATE TABLE recipes (id SERIAL NOT NULL PRIMARY KEY,title CHARACTER VARYING, yield_amount FLOAT, yield_amount_offset FLOAT, yield_type_id INTEGER DEFAULT '-1', instructions TEXT, photo TEXT, prep_time TIME);" ); + copyQuery = database->exec( "SELECT id,title,persons,instructions,photo,prep_time FROM recipes_copy" ); + if ( copyQuery.isActive() ) { + while ( copyQuery.next() ) { + TQSqlQuery query(TQString::null,database); + query.prepare( "INSERT INTO recipes VALUES (?, ?, ?, ?, ?, ?, ?, ?)" ); + query.addBindValue( copyQuery.value( 0 ) ); //id + query.addBindValue( copyQuery.value( 1 ) ); //title + query.addBindValue( copyQuery.value( 2 ) ); //persons, now yield_amount + query.addBindValue( 0 ); //yield_amount_offset + query.addBindValue( -1 ); //yield_type_id + query.addBindValue( copyQuery.value( 3 ) ); //instructions + query.addBindValue( copyQuery.value( 4 ) ); //photo + query.addBindValue( copyQuery.value( 5 ) ); //prep_time + query.exec(); + + emit progress(); + } + } + database->exec( "DROP TABLE recipes_copy" ); + + database->exec( "UPDATE db_info SET ver='0.82',generated_by='Krecipes SVN (20050902)';" ); + + if ( !database->commit() ) + kdDebug()<<"Update to 0.82 failed. Maybe you should try again."<<endl; + } + + if ( tqRound(version*100) < 83 ) { + database->transaction(); + + //====add a id columns to 'ingredient_list' to identify it for the prep method list + database->exec( "ALTER TABLE ingredient_list RENAME TO ingredient_list_copy;" ); + + database->exec( "CREATE TABLE ingredient_list (id SERIAL NOT NULL PRIMARY KEY, recipe_id INTEGER, ingredient_id INTEGER, amount FLOAT, amount_offset FLOAT, unit_id INTEGER, order_index INTEGER, group_id INTEGER);" ); + + TQSqlQuery copyQuery = database->exec( "SELECT recipe_id,ingredient_id,amount,amount_offset,unit_id,prep_method_id,order_index,group_id FROM ingredient_list_copy" ); + if ( copyQuery.isActive() ) { + while ( copyQuery.next() ) { + int ing_list_id = getNextInsertID("ingredient_list","id"); + + TQSqlQuery query(TQString::null,database); + query.prepare( "INSERT INTO ingredient_list VALUES (?, ?, ?, ?, ?, ?, ?, ?)" ); + query.addBindValue( ing_list_id ); + query.addBindValue( copyQuery.value( 0 ) ); + query.addBindValue( copyQuery.value( 1 ) ); + query.addBindValue( copyQuery.value( 2 ) ); + query.addBindValue( copyQuery.value( 3 ) ); + query.addBindValue( copyQuery.value( 4 ) ); + query.addBindValue( copyQuery.value( 6 ) ); + query.addBindValue( copyQuery.value( 7 ) ); + query.exec(); + + int prep_method_id = copyQuery.value( 5 ).toInt(); + if ( prep_method_id != -1 ) { + query.prepare( "INSERT INTO prep_method_list VALUES (?, ?, ?);" ); + query.addBindValue( ing_list_id ); + query.addBindValue( prep_method_id ); + query.addBindValue( 1 ); + query.exec(); + } + + emit progress(); + } + } + database->exec( "DROP TABLE ingredient_list_copy" ); + + database->exec( "CREATE INDEX ridil_index ON ingredient_list USING BTREE (recipe_id);" ); + database->exec( "CREATE INDEX iidil_index ON ingredient_list USING BTREE (ingredient_id);" ); + + database->exec( "UPDATE db_info SET ver='0.83',generated_by='Krecipes SVN (20050909)';" ); + + if ( !database->commit() ) { + kdDebug()<<"Update to 0.83 failed. Maybe you should try again."<<endl; + return; + } + } + + if ( tqRound(version*100) < 84 ) { + database->transaction(); + + database->exec( "ALTER TABLE recipes ADD COLUMN ctime TIMESTAMP" ); + database->exec( "ALTER TABLE recipes ADD COLUMN mtime TIMESTAMP" ); + database->exec( "ALTER TABLE recipes ADD COLUMN atime TIMESTAMP" ); + + database->exec( "UPDATE recipes SET ctime=CURRENT_TIMESTAMP, mtime=CURRENT_TIMESTAMP, atime=CURRENT_TIMESTAMP;" ); + + database->exec( "UPDATE db_info SET ver='0.84',generated_by='Krecipes SVN (20050913)';" ); + + if ( !database->commit() ) { + kdDebug()<<"Update to 0.84 failed. Maybe you should try again."<<endl; + return; + } + } + + if ( tqRound(version*100) < 85 ) { //this change altered the photo format, but this backend already used the newer format + database->transaction(); + + database->exec( "UPDATE db_info SET ver='0.85',generated_by='Krecipes SVN (20050926)';" ); + + if ( !database->commit() ) { + kdDebug()<<"Update to 0.85 failed. Maybe you should try again."<<endl; + return; + } + } + + if ( tqRound(version*100) < 86 ) { + database->transaction(); + + database->exec( "CREATE INDEX gidil_index ON ingredient_list USING BTREE (group_id);" ); + + TQSqlQuery query( "SELECT id,name FROM ingredient_groups ORDER BY name", database ); + + TQString last; + int lastID; + if ( query.isActive() ) { + while ( query.next() ) { + TQString name = query.value(1).toString(); + int id = query.value(0).toInt(); + if ( last == name ) { + TQString command = TQString("UPDATE ingredient_list SET group_id=%1 WHERE group_id=%2").arg(lastID).arg(id); + database->exec(command); + + command = TQString("DELETE FROM ingredient_groups WHERE id=%1").arg(id); + database->exec(command); + } + last = name; + lastID = id; + + emit progress(); + } + } + + database->exec( "UPDATE db_info SET ver='0.86',generated_by='Krecipes SVN (20050928)';" ); + if ( !database->commit() ) + kdDebug()<<"Update to 0.86 failed. Maybe you should try again."<<endl; + } + + if ( tqRound(version*100) < 87 ) { + //Load this default data so the user knows what rating criteria is + database->exec( TQString("INSERT INTO rating_criteria VALUES (1,'%1')").arg(i18n("Overall")) ); + database->exec( TQString("INSERT INTO rating_criteria VALUES (2,'%1')").arg(i18n("Taste") ) ); + database->exec( TQString("INSERT INTO rating_criteria VALUES (3,'%1')").arg(i18n("Appearance") ) ); + database->exec( TQString("INSERT INTO rating_criteria VALUES (4,'%1')").arg(i18n("Originality") ) ); + database->exec( TQString("INSERT INTO rating_criteria VALUES (5,'%1')").arg(i18n("Ease of Preparation") ) ); + + database->exec( "UPDATE db_info SET ver='0.87',generated_by='Krecipes SVN (20051014)'" ); + } + + if ( tqRound(version*100) < 90 ) { + database->exec("UPDATE db_info SET ver='0.9',generated_by='Krecipes 0.9'"); + } + + if ( tqRound(version*100) < 91 ) { + database->exec("CREATE index parent_id_index ON categories USING BTREE(parent_id)"); + database->exec("UPDATE db_info SET ver='0.91',generated_by='Krecipes SVN (20060526)'"); + } + + if ( tqRound(version*100) < 92 ) { + database->transaction(); + + //==================add a columns to 'units' to allow unit abbreviations + database->exec( "ALTER TABLE units RENAME TO units_copy" ); + + int nextval = -1; + TQSqlQuery getID( "SELECT nextval('units_id_seq')", database ); + if ( getID.isActive() && getID.first() ) + nextval = getID.value( 0 ).toInt(); + if ( nextval == -1 ) + kdDebug() << "Database update failed! Unable to update units sequence." << endl; + + database->exec( "CREATE TABLE units (id SERIAL NOT NULL PRIMARY KEY, name CHARACTER VARYING, name_abbrev CHARACTER VARYING, plural CHARACTER VARYING, plural_abbrev CHARACTER VARYING )" ); + TQSqlQuery copyQuery = database->exec( "SELECT id,name,plural FROM units_copy" ); + if ( copyQuery.isActive() ) { + while ( copyQuery.next() ) { + TQSqlQuery query(TQString::null,database); + query.prepare( "INSERT INTO units VALUES(?, ?, ?, ?, ?)" ); + query.addBindValue( copyQuery.value( 0 ) ); + query.addBindValue( copyQuery.value( 1 ) ); + query.addBindValue( TQVariant() ); + query.addBindValue( copyQuery.value( 2 ) ); + query.addBindValue( TQVariant() ); + query.exec(); + + emit progress(); + } + } + database->exec( "DROP TABLE units_copy" ); + + database->exec( "ALTER TABLE units_id_seq1 RENAME TO units_id_seq" ); + database->exec( "ALTER SEQUENCE units_id_seq RESTART WITH "+TQString::number(nextval) ); + + database->exec("UPDATE db_info SET ver='0.92',generated_by='Krecipes SVN (20060609)'"); + if ( !database->commit() ) + kdDebug()<<"Update to 0.92 failed. Maybe you should try again."<<endl; + } + + if ( tqRound(version*100) < 93 ) { + database->transaction(); + + database->exec( "ALTER TABLE ingredient_list ADD COLUMN substitute_for INTEGER" ); + + database->exec("UPDATE db_info SET ver='0.93',generated_by='Krecipes SVN (20060616)'"); + if ( !database->commit() ) + kdDebug()<<"Update to 0.93 failed. Maybe you should try again."<<endl; + } + + if ( tqRound(version*100) < 94 ) { + database->transaction(); + + database->exec( "ALTER TABLE units ADD COLUMN type INTEGER NOT NULL DEFAULT '0'" ); + + database->exec("UPDATE db_info SET ver='0.94',generated_by='Krecipes SVN (20060712)'"); + if ( !database->commit() ) + kdDebug()<<"Update to 0.94 failed. Maybe you should try again."<<endl; + } + + if ( tqRound(version*100) < 95 ) { + database->exec( "DROP TABLE ingredient_weights" ); + createTable( "ingredient_weights" ); + database->exec( "UPDATE db_info SET ver='0.95',generated_by='Krecipes SVN (20060726)'" ); + } +} + +void PSqlRecipeDB::addColumn( const TQString &new_table_sql, const TQString &new_col_info, const TQString &default_value, const TQString &table_name, int col_index ) +{ + TQString command; + + command = TQString(new_table_sql).arg(table_name+"_copy").arg(TQString::null); + kdDebug()<<"calling: "<<command<<endl; + TQSqlQuery query( command, database ); + + command = "SELECT * FROM "+table_name+";"; + query.exec( command ); + if ( query.isActive() ) { + while ( query.next() ) { + TQStringList dataList; + for ( int i = 0 ;; ++i ) { + TQVariant variant = query.value(i); + if ( variant.type() == TQVariant::Invalid ) break; + + dataList << "'"+variant.toString()+"'"; + } + command = "INSERT INTO "+table_name+"_copy VALUES("+dataList.join(",")+");"; + kdDebug()<<"calling: "<<command<<endl; + TQSqlQuery insert_query( command, database ); + + emit progress(); + } + } + query.exec( "DROP TABLE "+table_name+";" ); + query.exec( TQString(new_table_sql).arg(table_name).arg(new_col_info+",") ); + query.exec( "SELECT * FROM "+table_name+"_copy;" ); + if ( query.isActive() ) { + while ( query.next() ) { + TQStringList dataList; + for ( int i = 0 ;; ++i ) { + if ( i == col_index ) + dataList << default_value; + + TQVariant variant = query.value(i); + if ( variant.type() == TQVariant::Invalid ) break; + + dataList << "'"+variant.toString()+"'"; + } + command = "INSERT INTO "+table_name+" VALUES(" +dataList.join(",")+");"; + TQSqlQuery insert_query( command, database ); + kdDebug()<<"calling: "<<command<<endl; + + emit progress(); + } + } + query.exec( "DROP TABLE "+table_name+"_copy;" ); +} + +int PSqlRecipeDB::lastInsertID() +{ + return last_insert_id; +} + +int PSqlRecipeDB::getNextInsertID( const TQString &table, const TQString &column ) +{ + TQString command = TQString( "SELECT nextval('%1_%2_seq');" ).arg( table ).arg( column ); + TQSqlQuery getID( command, database ); + + if ( getID.isActive() && getID.first() ) { + last_insert_id = getID.value( 0 ).toInt(); + } + else + last_insert_id = -1; + + return last_insert_id; +} + +void PSqlRecipeDB::givePermissions( const TQString & /*dbName*/, const TQString &username, const TQString &password, const TQString & /*clientHost*/ ) +{ + TQStringList tables; + tables << "ingredient_info" << "ingredient_list" << "ingredient_properties" << "ingredients" << "recipes" << "unit_list" << "units" << "units_conversion" << "categories" << "category_list" << "authors" << "author_list" << "prep_methods" << "db_info" << "ingredient_groups" << "ingredient_weights" << "prep_method_list" << "yield_types" << "ratings" << "rating_criteria" << "rating_criterion_list"; + + //we also have to grant permissions on the sequences created + tables << "authors_id_seq" << "categories_id_seq" << "ingredient_properties_id_seq" << "ingredient_weights_id_seq" << "ingredients_id_seq" << "prep_methods_id_seq" << "recipes_id_seq" << "units_id_seq" << "ingredient_groups_id_seq" << "yield_types_id_seq" << "ingredient_list_id_seq" << "ratings_id_seq" << "rating_criteria_id_seq"; + + TQString command; + + kdDebug() << "I'm doing the query to create the new user" << endl; + command = "CREATE USER " + username; + if ( !password.isEmpty() ) + command.append( "WITH PASSWORD '" + password + "'" ); + command.append( ";" ); + TQSqlQuery permissionsToSet( command, database ); + + kdDebug() << "I'm doing the query to setup permissions\n"; + command = TQString( "GRANT ALL ON %1 TO %2;" ).arg( tables.join( "," ) ).arg( username ); + permissionsToSet.exec( command ); +} + +void PSqlRecipeDB::empty( void ) +{ + TQSqlRecipeDB::empty(); + + TQStringList tables; + tables << "authors_id_seq" << "categories_id_seq" << "ingredient_properties_id_seq" << "ingredient_weights_id_seq" << "ingredients_id_seq" << "prep_methods_id_seq" << "recipes_id_seq" << "units_id_seq" << "ingredient_groups_id_seq" << "yield_types_id_seq" << "ingredient_list_id_seq" << "prep_method_list_id_seq" << "ratings_id_seq" << "rating_criteria_id_seq"; + + TQSqlQuery tablesToEmpty( TQString::null, database ); + for ( TQStringList::Iterator it = tables.begin(); it != tables.end(); ++it ) { + TQString command = TQString( "DELETE FROM %1;" ).arg( *it ); + tablesToEmpty.exec( command ); + } +} + +#include "psqlrecipedb.moc" diff --git a/src/backends/PostgreSQL/psqlrecipedb.h b/src/backends/PostgreSQL/psqlrecipedb.h new file mode 100644 index 0000000..feaee9a --- /dev/null +++ b/src/backends/PostgreSQL/psqlrecipedb.h @@ -0,0 +1,64 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + + +#ifndef PSQLRECIPEDB_H +#define PSQLRECIPEDB_H + +#define PSQL_DRIVER "TQPSQL7" + +#include "qsqlrecipedb.h" + +/** +@author Jason Kivlighn +*/ +class PSqlRecipeDB : public TQSqlRecipeDB +{ + + TQ_OBJECT + +private: + void createDB( void ); + +public: + PSqlRecipeDB( const TQString& host, const TQString& user = TQString::null, const TQString& pass = TQString::null, const TQString& DBName = DEFAULT_DB_NAME, int port = 0 ); + ~PSqlRecipeDB( void ); + + int lastInsertID(); + void initializeData(); + + void createTable( const TQString &tableName ); + void givePermissions( const TQString &dbName, const TQString &username, const TQString &password, const TQString &clientHost ); + +protected: + virtual TQString qsqlDriverPlugin() const + { + return PSQL_DRIVER; + } + virtual int getNextInsertID( const TQString &table, const TQString &column ); + + virtual void empty( void ); + +private: + void portOldDatabases( float version ); + TQStringList backupCommand() const; + TQStringList restoreCommand() const; + + void addColumn( const TQString &new_table_sql, const TQString &new_col_info, const TQString &default_value, const TQString &table_name, int col_index ); + + int last_insert_id; +}; + + + + +#endif diff --git a/src/backends/SQLite/Makefile.am b/src/backends/SQLite/Makefile.am new file mode 100644 index 0000000..e136dd1 --- /dev/null +++ b/src/backends/SQLite/Makefile.am @@ -0,0 +1,15 @@ +## Makefile.am for krecipes + +# this is the program that gets installed. it's name is used for all +# of the other Makefile.am variables + +# set the include path for X, tqt and TDE +INCLUDES = -I$(srcdir) -I$(srcdir)/.. -I$(srcdir)/../.. $(all_includes) + +# Instructions for building the convenience library +noinst_LTLIBRARIES=libkrecsqlite.la +libkrecsqlite_la_SOURCES=literecipedb.cpp +libkrecsqlite_la_METASOURCES=AUTO + +#the library search path. +libkrecsqlite_la_LDFLAGS = $(KDE_RPATH) $(all_libraries) diff --git a/src/backends/SQLite/literecipedb.cpp b/src/backends/SQLite/literecipedb.cpp new file mode 100644 index 0000000..9531791 --- /dev/null +++ b/src/backends/SQLite/literecipedb.cpp @@ -0,0 +1,1033 @@ +/*************************************************************************** +* Copyright (C) 2003-2004 by * +* Unai Garro ([email protected]) * +* Jason Kivlighn ([email protected]) * +* Cyril Bosselut ([email protected]) * +* * +* Copyright (C) 2006 Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "literecipedb.h" + +#include <tqbuffer.h> + +#include <kdebug.h> +#include <tdeconfig.h> +#include <tdeglobal.h> +#include <tdelocale.h> + +#ifdef HAVE_CONFIG_H +#include "config.h" +#endif + +#if HAVE_SQLITE3 +#include <sqlite3.h> +#elif HAVE_SQLITE +#include <sqlite.h> +#endif + +//keep these two around for porting old databases +int sqlite_decode_binary( const unsigned char *in, unsigned char *out ); +TQString escape( const TQString &s ); + +LiteRecipeDB::LiteRecipeDB( const TQString &_dbFile ) : TQSqlRecipeDB( TQString::null, TQString::null, TQString::null, _dbFile ) +{ +/* TDEConfig * config = TDEGlobal::config(); + config->setGroup( "Server" ); + + if ( dbFile.isNull() ) + dbFile = config->readEntry( "DBFile", locateLocal ( "appdata", DB_FILENAME ) ); +*/ +} + +LiteRecipeDB::~LiteRecipeDB() +{ +} + +int LiteRecipeDB::lastInsertID() +{ + int lastID = -1; + TQSqlQuery query("SELECT lastInsertID()",database); + if ( query.isActive() && query.first() ) + lastID = query.value(0).toInt(); + + //kdDebug()<<"lastInsertID(): "<<lastID<<endl; + + return lastID; +} + +TQStringList LiteRecipeDB::backupCommand() const +{ + #if HAVE_SQLITE + TQString binary = "sqlite"; + #elif HAVE_SQLITE3 + TQString binary = "sqlite3"; + #endif + + TDEConfig * config = TDEGlobal::config(); + config->setGroup( "Server" ); + binary = config->readEntry( "SQLitePath", binary ); + + TQStringList command; + command<<binary<<database->databaseName()<<".dump"; + return command; +} + +TQStringList LiteRecipeDB::restoreCommand() const +{ + #if HAVE_SQLITE + TQString binary = "sqlite"; + #elif HAVE_SQLITE3 + TQString binary = "sqlite3"; + #endif + + TDEConfig * config = TDEGlobal::config(); + config->setGroup( "Server" ); + binary = config->readEntry( "SQLitePath", binary ); + + TQStringList command; + command<<binary<<database->databaseName(); + return command; +} + +void LiteRecipeDB::createDB() +{ + //The file is created by SQLite automatically +} + +void LiteRecipeDB::createTable( const TQString &tableName ) +{ + + TQStringList commands; + + if ( tableName == "recipes" ) + commands << TQString( "CREATE TABLE recipes (id INTEGER NOT NULL,title VARCHAR(%1), yield_amount FLOAT, yield_amount_offset FLOAT, yield_type_id INTEGER DEFAULT '-1', instructions TEXT, photo BLOB, prep_time TIME, ctime TIMESTAMP, mtime TIMESTAMP, atime TIMESTAMP, PRIMARY KEY (id));" ).arg( maxRecipeTitleLength() ); + + else if ( tableName == "ingredients" ) + commands << TQString( "CREATE TABLE ingredients (id INTEGER NOT NULL, name VARCHAR(%1), PRIMARY KEY (id));" ).arg( maxIngredientNameLength() ); + + else if ( tableName == "ingredient_list" ) { + commands << "CREATE TABLE ingredient_list (id INTEGER NOT NULL, recipe_id INTEGER, ingredient_id INTEGER, amount FLOAT, amount_offset FLOAT, unit_id INTEGER, order_index INTEGER, group_id INTEGER, substitute_for INTEGER, PRIMARY KEY(id) );" + << "CREATE index ridil_index ON ingredient_list(recipe_id);" + << "CREATE index iidil_index ON ingredient_list(ingredient_id);" + << "CREATE index gidil_index ON ingredient_list(group_id);"; + } + + else if ( tableName == "unit_list" ) + commands << "CREATE TABLE unit_list (ingredient_id INTEGER, unit_id INTEGER);"; + + else if ( tableName == "units" ) + commands << TQString( "CREATE TABLE units (id INTEGER NOT NULL, name VARCHAR(%1), name_abbrev VARCHAR(%2), plural VARCHAR(%3), plural_abbrev VARCHAR(%4), type INTEGER NOT NULL DEFAULT '0',PRIMARY KEY (id));" ) + .arg( maxUnitNameLength() ).arg( maxUnitNameLength() ).arg( maxUnitNameLength() ).arg( maxUnitNameLength() ); + + else if ( tableName == "prep_methods" ) + commands << TQString( "CREATE TABLE prep_methods (id INTEGER NOT NULL, name VARCHAR(%1), PRIMARY KEY (id));" ).arg( maxPrepMethodNameLength() ); + + else if ( tableName == "prep_method_list" ) { + commands << "CREATE TABLE prep_method_list (ingredient_list_id INTEGER NOT NULL,prep_method_id INTEGER NOT NULL, order_index INTEGER );" + << "CREATE index iid_index ON prep_method_list(ingredient_list_id);" + << "CREATE index pid_index ON prep_method_list(prep_method_id);"; + } + + else if ( tableName == "ingredient_info" ) + commands << "CREATE TABLE ingredient_info (ingredient_id INTEGER, property_id INTEGER, amount FLOAT, per_units INTEGER);"; + + else if ( tableName == "ingredient_properties" ) + commands << "CREATE TABLE ingredient_properties (id INTEGER NOT NULL,name VARCHAR(20), units VARCHAR(20), PRIMARY KEY (id));"; + + else if ( tableName == "ingredient_weights" ) { + commands << "CREATE TABLE ingredient_weights (id INTEGER NOT NULL, ingredient_id INTEGER NOT NULL, amount FLOAT, unit_id INTEGER, weight FLOAT, weight_unit_id INTEGER, prep_method_id INTEGER, PRIMARY KEY (id) );" + + << "CREATE index weight_wid_index ON ingredient_weights(weight_unit_id)" + << "CREATE index weight_pid_index ON ingredient_weights(prep_method_id)" + << "CREATE index weight_uid_index ON ingredient_weights(unit_id)" + << "CREATE index weight_iid_index ON ingredient_weights(ingredient_id)"; + } + + else if ( tableName == "units_conversion" ) + commands << "CREATE TABLE units_conversion (unit1_id INTEGER, unit2_id INTEGER, ratio FLOAT);"; + + else if ( tableName == "categories" ) { + commands << TQString( "CREATE TABLE categories (id INTEGER NOT NULL, name varchar(%1) default NULL, parent_id INGEGER NOT NULL default -1, PRIMARY KEY (id));" ).arg( maxCategoryNameLength() ); + commands << "CREATE index parent_id_index ON categories(parent_id);"; + } + else if ( tableName == "category_list" ) { + commands << "CREATE TABLE category_list (recipe_id INTEGER NOT NULL,category_id INTEGER NOT NULL);" + << "CREATE index rid_index ON category_list(recipe_id);" + << "CREATE index cid_index ON category_list(category_id);"; + } + + else if ( tableName == "authors" ) + commands << TQString( "CREATE TABLE authors (id INTEGER NOT NULL, name varchar(%1) default NULL,PRIMARY KEY (id));" ).arg( maxAuthorNameLength() ); + + else if ( tableName == "author_list" ) + commands << "CREATE TABLE author_list (recipe_id INTEGER NOT NULL,author_id INTEGER NOT NULL);"; + + else if ( tableName == "db_info" ) { + commands << "CREATE TABLE db_info (ver FLOAT NOT NULL,generated_by varchar(200) default NULL);"; + commands << TQString( "INSERT INTO db_info VALUES(%1,'Krecipes %2');" ).arg( latestDBVersion() ).arg( krecipes_version() ); + } + else if ( tableName == "ingredient_groups" ) { + commands << TQString( "CREATE TABLE ingredient_groups (id INTEGER NOT NULL, name varchar(%1), PRIMARY KEY (id));" ).arg( maxIngGroupNameLength() ); + } + else if ( tableName == "yield_types" ) { + commands << TQString( "CREATE TABLE yield_types (id INTEGER NOT NULL, name varchar(%1), PRIMARY KEY (id));" ).arg( maxYieldTypeLength() ); + } + + else if ( tableName == "ratings" ) + commands << "CREATE TABLE ratings (id INTEGER NOT NULL, recipe_id int(11) NOT NULL, comment TEXT, rater TEXT, created TIMESTAMP, PRIMARY KEY (id));"; + + else if ( tableName == "rating_criteria" ) + commands << "CREATE TABLE rating_criteria (id INTEGER NOT NULL, name TEXT, PRIMARY KEY (id));"; + + else if ( tableName == "rating_criterion_list" ) + commands << "CREATE TABLE rating_criterion_list (rating_id INTEGER NOT NULL, rating_criterion_id INTEGER, stars FLOAT);"; + + else + return ; + + // execute the queries + for ( TQStringList::const_iterator it = commands.begin(); it != commands.end(); ++it ) + database->exec( *it ); + +} + +void LiteRecipeDB::portOldDatabases( float version ) +{ + TQString command; + if ( tqRound(version*10) < 5 ) { + //===========add prep_method_id to ingredient_list table + //There's no ALTER command in SQLite, so we have to copy all data to a new table and then recreate the table with the prep_method_id + database->exec( "CREATE TABLE ingredient_list_copy (recipe_id INTEGER, ingredient_id INTEGER, amount FLOAT, unit_id INTEGER, order_index INTEGER);" ); + TQSqlQuery copyQuery( "SELECT recipe_id,ingredient_id,amount,unit_id,order_index FROM ingredient_list;", database ); + if ( copyQuery.isActive() ) { + while ( copyQuery.next() ) { + command = TQString( "INSERT INTO ingredient_list_copy VALUES(%1,%2,%3,%4,%5);" ) + .arg( copyQuery.value( 0 ).toInt() ) + .arg( copyQuery.value( 1 ).toInt() ) + .arg( copyQuery.value( 2 ).toDouble() ) + .arg( copyQuery.value( 3 ).toInt() ) + .arg( copyQuery.value( 4 ).toInt() ); + database->exec( command ); + + emit progress(); + } + } + database->exec( "DROP TABLE ingredient_list" ); + database->exec( "CREATE TABLE ingredient_list (recipe_id INTEGER, ingredient_id INTEGER, amount FLOAT, unit_id INTEGER, prep_method_id INTEGER, order_index INTEGER);" ); + copyQuery = database->exec( "SELECT * FROM ingredient_list_copy" ); + if ( copyQuery.isActive() ) { + while ( copyQuery.next() ) { + command = TQString( "INSERT INTO ingredient_list VALUES(%1,%2,%3,%4,%5,%6);" ) + .arg( copyQuery.value( 0 ).toInt() ) + .arg( copyQuery.value( 1 ).toInt() ) + .arg( copyQuery.value( 2 ).toDouble() ) + .arg( copyQuery.value( 3 ).toInt() ) + .arg( -1 ) //default prep method + .arg( copyQuery.value( 4 ).toInt() ); + database->exec( command ); + + emit progress(); + } + } + database->exec( "DROP TABLE ingredient_list_copy" ); + + database->exec( "CREATE index ridil_index ON ingredient_list(recipe_id);" ); + database->exec( "CREATE index iidil_index ON ingredient_list(ingredient_id);" ); + + + //==============expand length of author name to 50 characters + database->exec( "CREATE TABLE authors_copy (id INTEGER, name varchar(20));" ); + copyQuery = database->exec( "SELECT * FROM authors;" ); + if ( copyQuery.isActive() ) { + + while ( copyQuery.next() ) { + command = TQString( "INSERT INTO authors_copy VALUES(%1,'%2');" ) + .arg( copyQuery.value( 0 ).toInt() ) + .arg( escape( copyQuery.value( 1 ).toString() ) ); + database->exec( command ); + + emit progress(); + } + } + database->exec( "DROP TABLE authors" ); + database->exec( "CREATE TABLE authors (id INTEGER NOT NULL, name varchar(50) default NULL,PRIMARY KEY (id));" ); + copyQuery = database->exec( "SELECT * FROM authors_copy" ); + if ( copyQuery.isActive() ) { + + while ( copyQuery.next() ) { + command = TQString( "INSERT INTO authors VALUES(%1,'%2');" ) + .arg( copyQuery.value( 0 ).toInt() ) + .arg( escape( copyQuery.value( 1 ).toString() ) ); + database->exec( command ); + + emit progress(); + } + } + database->exec( "DROP TABLE authors_copy" ); + + + //==================expand length of category name to 40 characters + database->exec( "CREATE TABLE categories_copy (id INTEGER, name varchar(20));" ); + copyQuery = database->exec( "SELECT * FROM categories;" ); + if ( copyQuery.isActive() ) { + + while ( copyQuery.next() ) { + command = TQString( "INSERT INTO categories_copy VALUES(%1,'%2');" ) + .arg( copyQuery.value( 0 ).toInt() ) + .arg( escape( copyQuery.value( 1 ).toString() ) ); + database->exec( command ); + + emit progress(); + } + } + database->exec( "DROP TABLE categories" ); + database->exec( "CREATE TABLE categories (id INTEGER NOT NULL, name varchar(40) default NULL,PRIMARY KEY (id));" ); + copyQuery = database->exec( "SELECT * FROM categories_copy" ); + if ( copyQuery.isActive() ) { + + while ( copyQuery.next() ) { + command = TQString( "INSERT INTO categories VALUES(%1,'%2');" ) + .arg( copyQuery.value( 0 ).toInt() ) + .arg( escape( copyQuery.value( 1 ).toString() ) ); + database->exec( command ); + + emit progress(); + } + } + database->exec( "DROP TABLE categories_copy" ); + + //================Set the version to the new one (0.5) + command = "DELETE FROM db_info;"; // Remove previous version records if they exist + database->exec( command ); + command = "INSERT INTO db_info VALUES(0.5,'Krecipes 0.5');"; + database->exec( command ); + } + + if ( tqRound(version*10) < 6 ) { + //==================add a column to 'categories' to allow subcategories + database->exec( "CREATE TABLE categories_copy (id INTEGER, name varchar(40));" ); + TQSqlQuery copyQuery = database->exec( "SELECT * FROM categories;" ); + if ( copyQuery.isActive() ) { + + while ( copyQuery.next() ) { + command = TQString( "INSERT INTO categories_copy VALUES(%1,'%2');" ) + .arg( copyQuery.value( 0 ).toInt() ) + .arg( escape( copyQuery.value( 1 ).toString() ) ); + database->exec( command ); + + emit progress(); + } + } + database->exec( "DROP TABLE categories" ); + database->exec( "CREATE TABLE categories (id INTEGER NOT NULL, name varchar(40) default NULL, parent_id INTEGER NOT NULL, PRIMARY KEY (id));" ); + copyQuery = database->exec( "SELECT * FROM categories_copy" ); + if ( copyQuery.isActive() ) { + + while ( copyQuery.next() ) { + command = TQString( "INSERT INTO categories VALUES(%1,'%2',-1);" ) + .arg( copyQuery.value( 0 ).toInt() ) + .arg( escape( copyQuery.value( 1 ).toString() ) ); + database->exec( command ); + + emit progress(); + } + } + database->exec( "DROP TABLE categories_copy" ); + + //================Set the version to the new one (0.6) + command = "DELETE FROM db_info;"; // Remove previous version records if they exist + database->exec( command ); + command = "INSERT INTO db_info VALUES(0.6,'Krecipes 0.6');"; + database->exec( command ); + } + + if ( tqRound(version*100) < 61 ) { + //==================add a column to 'recipes' to allow prep time + database->exec( "CREATE TABLE recipes_copy (id INTEGER NOT NULL,title VARCHAR(200),persons INTEGER,instructions TEXT, photo BLOB, PRIMARY KEY (id));" ); + TQSqlQuery copyQuery = database->exec( "SELECT * FROM recipes;" ); + if ( copyQuery.isActive() ) { + + while ( copyQuery.next() ) { + command = TQString( "INSERT INTO recipes_copy VALUES(%1,'%2','%3','%4','%5');" ) + .arg( escape( copyQuery.value( 0 ).toString() ) ) + .arg( escape( copyQuery.value( 1 ).toString() ) ) + .arg( escape( copyQuery.value( 2 ).toString() ) ) + .arg( escape( copyQuery.value( 3 ).toString() ) ) + .arg( escape( copyQuery.value( 4 ).toString() ) ); + database->exec( command ); + + emit progress(); + } + } + database->exec( "DROP TABLE recipes" ); + database->exec( "CREATE TABLE recipes (id INTEGER NOT NULL,title VARCHAR(200),persons INTEGER,instructions TEXT, photo BLOB, prep_time TIME, PRIMARY KEY (id));" ); + copyQuery = database->exec( "SELECT * FROM recipes_copy" ); + if ( copyQuery.isActive() ) { + + while ( copyQuery.next() ) { + command = TQString( "INSERT INTO recipes VALUES(%1,'%2','%3','%4','%5',NULL);" ) + .arg( escape( copyQuery.value( 0 ).toString() ) ) + .arg( escape( copyQuery.value( 1 ).toString() ) ) + .arg( escape( copyQuery.value( 2 ).toString() ) ) + .arg( escape( copyQuery.value( 3 ).toString() ) ) + .arg( escape( copyQuery.value( 4 ).toString() ) ); + + database->exec( command ); + + emit progress(); + } + } + database->exec( "DROP TABLE recipes_copy" ); + + //================Set the version to the new one (0.61) + command = "DELETE FROM db_info;"; // Remove previous version records if they exist + database->exec( command ); + command = "INSERT INTO db_info VALUES(0.61,'Krecipes 0.6');"; + database->exec( command ); + } + + if ( tqRound(version*100) < 62 ) { + database->transaction(); + + //==================add a column to 'ingredient_list' to allow grouping ingredients + database->exec( "CREATE TABLE ingredient_list_copy (recipe_id INTEGER, ingredient_id INTEGER, amount FLOAT, unit_id INTEGER, prep_method_id INTEGER, order_index INTEGER);" ); + TQSqlQuery copyQuery = database->exec( "SELECT * FROM ingredient_list;" ); + if ( copyQuery.isActive() ) { + + while ( copyQuery.next() ) { + command = "INSERT INTO ingredient_list_copy VALUES('" + escape( copyQuery.value( 0 ).toString() ) + + "','" + escape( copyQuery.value( 1 ).toString() ) + + "','" + escape( copyQuery.value( 2 ).toString() ) + + "','" + escape( copyQuery.value( 3 ).toString() ) + + "','" + escape( copyQuery.value( 4 ).toString() ) + + "','" + escape( copyQuery.value( 5 ).toString() ) + + "');"; + database->exec( command ); + + emit progress(); + } + } + database->exec( "DROP TABLE ingredient_list" ); + database->exec( "CREATE TABLE ingredient_list (recipe_id INTEGER, ingredient_id INTEGER, amount FLOAT, unit_id INTEGER, prep_method_id INTEGER, order_index INTEGER, group_id INTEGER);" ); + copyQuery = database->exec( "SELECT * FROM ingredient_list_copy" ); + if ( copyQuery.isActive() ) { + + while ( copyQuery.next() ) { + command = "INSERT INTO ingredient_list VALUES('" + escape( copyQuery.value( 0 ).toString() ) + + "','" + escape( copyQuery.value( 1 ).toString() ) + + "','" + escape( copyQuery.value( 2 ).toString() ) + + "','" + escape( copyQuery.value( 3 ).toString() ) + + "','" + escape( copyQuery.value( 4 ).toString() ) + + "','" + escape( copyQuery.value( 5 ).toString() ) + + "',-1)"; + database->exec( command ); + + emit progress(); + } + } + database->exec( "DROP TABLE ingredient_list_copy" ); + + command = "DELETE FROM db_info;"; // Remove previous version records if they exist + database->exec( command ); + command = "INSERT INTO db_info VALUES(0.62,'Krecipes 0.7');"; + database->exec( command ); + + database->commit(); + } + + if ( tqRound(version*100) < 63 ) { + database->transaction(); + + //==================add a column to 'units' to allow handling plurals + database->exec( "CREATE TABLE units_copy (id INTEGER NOT NULL, name VARCHAR(20), PRIMARY KEY (id));" ); + TQSqlQuery copyQuery = database->exec( "SELECT id,name FROM units;" ); + if ( copyQuery.isActive() ) { + + while ( copyQuery.next() ) { + command = "INSERT INTO units_copy VALUES('" + escape( copyQuery.value( 0 ).toString() ) + + "','" + escape( copyQuery.value( 1 ).toString() ) + + "');"; + database->exec( command ); + + emit progress(); + } + } + database->exec( "DROP TABLE units" ); + database->exec( "CREATE TABLE units (id INTEGER NOT NULL, name VARCHAR(20), plural VARCHAR(20), PRIMARY KEY (id));" ); + copyQuery = database->exec( "SELECT id,name FROM units_copy" ); + if ( copyQuery.isActive() ) { + while ( copyQuery.next() ) { + command = "INSERT INTO units VALUES('" + escape( copyQuery.value( 0 ).toString() ) + + "','" + escape( copyQuery.value( 1 ).toString() ) + + "',NULL)"; + database->exec( command ); + + emit progress(); + } + } + database->exec( "DROP TABLE units_copy" ); + + TQSqlQuery result = database->exec( "SELECT id,name FROM units WHERE plural ISNULL;" ); + if ( result.isActive() ) { + while ( result.next() ) { + command = "UPDATE units SET plural='" + escape( result.value( 1 ).toString() ) + "' WHERE id=" + TQString::number( result.value( 0 ).toInt() ); + database->exec( command ); + + emit progress(); + } + } + + command = "DELETE FROM db_info;"; // Remove previous version records if they exist + database->exec( command ); + command = "INSERT INTO db_info VALUES(0.63,'Krecipes 0.7');"; + database->exec( command ); + + database->commit(); + } + + if ( tqRound(version*10) < 7 ) { //simply call 0.63 -> 0.7 + database->exec( "UPDATE db_info SET ver='0.7';" ); + } + + if ( tqRound(version*100) < 81 ) { + database->transaction(); + addColumn("CREATE TABLE %1 (recipe_id INTEGER, ingredient_id INTEGER, amount FLOAT, %2 unit_id INTEGER, prep_method_id INTEGER, order_index INTEGER, group_id INTEGER)","amount_offset FLOAT","'0'","ingredient_list",3); + + //addColumn() doesn't preserve indexes + database->exec("CREATE index ridil_index ON ingredient_list(recipe_id)"); + database->exec("CREATE index iidil_index ON ingredient_list(ingredient_id)"); + + database->exec( "UPDATE db_info SET ver='0.81',generated_by='Krecipes SVN (20050816)';" ); + database->commit(); + } + + if ( tqRound(version*100) < 82 ) { + database->transaction(); + + //==================add a columns to 'recipes' to allow yield range + yield type + database->exec( "CREATE TABLE recipes_copy (id INTEGER NOT NULL,title VARCHAR(200),persons INTEGER,instructions TEXT, photo BLOB, prep_time TIME, PRIMARY KEY (id));" ); + TQSqlQuery copyQuery = database->exec( "SELECT id,title,persons,instructions,photo,prep_time FROM recipes;" ); + if ( copyQuery.isActive() ) { + + while ( copyQuery.next() ) { + command = "INSERT INTO recipes_copy VALUES('" + + escape( copyQuery.value( 0 ).toString() ) //id + + "','" + escape( copyQuery.value( 1 ).toString() ) //title + + "','" + escape( copyQuery.value( 2 ).toString() ) //persons + + "','" + escape( copyQuery.value( 3 ).toString() ) //instructions + + "','" + escape( copyQuery.value( 4 ).toString() ) //photo + + "','" + escape( copyQuery.value( 5 ).toString() ) //prep_time + + "')"; + database->exec( command ); + + emit progress(); + } + } + database->exec( "DROP TABLE recipes" ); + database->exec( "CREATE TABLE recipes (id INTEGER NOT NULL,title VARCHAR(200), yield_amount FLOAT, yield_amount_offset FLOAT, yield_type_id INTEGER, instructions TEXT, photo BLOB, prep_time TIME, PRIMARY KEY (id));" ); + copyQuery = database->exec( "SELECT id,title,persons,instructions,photo,prep_time FROM recipes_copy" ); + if ( copyQuery.isActive() ) { + + while ( copyQuery.next() ) { + command = "INSERT INTO recipes VALUES('" + + escape( copyQuery.value( 0 ).toString() ) //id + + "','" + escape( copyQuery.value( 1 ).toString() ) //title + + "','" + escape( copyQuery.value( 2 ).toString() ) //persons, now yield_amount + + "','0" //yield_amount_offset + + "','-1" //yield_type_id + + "','" + escape( copyQuery.value( 3 ).toString() ) //instructions + + "','" + escape( copyQuery.value( 4 ).toString() ) //photo + + "','" + escape( copyQuery.value( 5 ).toString() ) //prep_time + + "')"; + database->exec( command ); + + emit progress(); + } + } + database->exec( "DROP TABLE recipes_copy" ); + + database->exec( "UPDATE db_info SET ver='0.82',generated_by='Krecipes SVN (20050902)';" ); + database->commit(); + } + + if ( tqRound(version*100) < 83 ) { + database->transaction(); + + //====add a id columns to 'ingredient_list' to identify it for the prep method list + database->exec( "CREATE TABLE ingredient_list_copy (recipe_id INTEGER, ingredient_id INTEGER, amount FLOAT, amount_offset FLOAT, unit_id INTEGER, prep_method_id INTEGER, order_index INTEGER, group_id INTEGER);" ); + TQSqlQuery copyQuery = database->exec( "SELECT recipe_id,ingredient_id,amount,amount_offset,unit_id,prep_method_id,order_index,group_id FROM ingredient_list" ); + if ( copyQuery.isActive() ) { + + while ( copyQuery.next() ) { + command = "INSERT INTO ingredient_list_copy VALUES('" + + escape( copyQuery.value( 0 ).toString() ) + + "','" + escape( copyQuery.value( 1 ).toString() ) + + "','" + escape( copyQuery.value( 2 ).toString() ) + + "','" + escape( copyQuery.value( 3 ).toString() ) + + "','" + escape( copyQuery.value( 4 ).toString() ) + + "','" + escape( copyQuery.value( 5 ).toString() ) + + "','" + escape( copyQuery.value( 6 ).toString() ) + + "','" + escape( copyQuery.value( 7 ).toString() ) + + "')"; + database->exec( command ); + + emit progress(); + } + } + database->exec( "DROP TABLE ingredient_list" ); + database->exec( "CREATE TABLE ingredient_list (id INTEGER NOT NULL, recipe_id INTEGER, ingredient_id INTEGER, amount FLOAT, amount_offset FLOAT, unit_id INTEGER, order_index INTEGER, group_id INTEGER, PRIMARY KEY(id) );" ); + + copyQuery = database->exec( "SELECT recipe_id,ingredient_id,amount,amount_offset,unit_id,prep_method_id,order_index,group_id FROM ingredient_list_copy" ); + if ( copyQuery.isActive() ) { + + while ( copyQuery.next() ) { + command = "INSERT INTO ingredient_list VALUES(" + + TQString("NULL") + + ",'" + escape( copyQuery.value( 0 ).toString() ) + + "','" + escape( copyQuery.value( 1 ).toString() ) + + "','" + escape( copyQuery.value( 2 ).toString() ) + + "','" + escape( copyQuery.value( 3 ).toString() ) + + "','" + escape( copyQuery.value( 4 ).toString() ) + + "','" + escape( copyQuery.value( 6 ).toString() ) + + "','" + escape( copyQuery.value( 7 ).toString() ) + + "')"; + database->exec( command ); + + int prep_method_id = copyQuery.value( 5 ).toInt(); + if ( prep_method_id != -1 ) { + command = "INSERT INTO prep_method_list VALUES('" + + TQString::number(lastInsertID()) + + "','" + TQString::number(prep_method_id) + + "','1" //order_index + + "')"; + database->exec( command ); + } + + emit progress(); + } + } + database->exec( "DROP TABLE ingredient_list_copy" ); + + database->exec( "CREATE INDEX ridil_index ON ingredient_list(recipe_id);" ); + database->exec( "CREATE INDEX iidil_index ON ingredient_list(ingredient_id);" ); + + database->exec( "UPDATE db_info SET ver='0.83',generated_by='Krecipes SVN (20050909)';" ); + + database->commit(); + } + + if ( tqRound(version*100) < 84 ) { + database->transaction(); + + //==================add a columns to 'recipes' to allow storing atime, mtime, ctime + database->exec( "CREATE TABLE recipes_copy (id INTEGER NOT NULL,title VARCHAR(200), yield_amount FLOAT, yield_amount_offset FLOAT, yield_type_id INTEGER, instructions TEXT, photo BLOB, prep_time TIME, PRIMARY KEY (id));" ); + TQSqlQuery copyQuery = database->exec( "SELECT id,title,yield_amount,yield_amount_offset,yield_type_id,instructions,photo,prep_time FROM recipes;" ); + if ( copyQuery.isActive() ) { + + while ( copyQuery.next() ) { + command = "INSERT INTO recipes_copy VALUES('" + + escape( copyQuery.value( 0 ).toString() ) + + "','" + escape( copyQuery.value( 1 ).toString() ) + + "','" + escape( copyQuery.value( 2 ).toString() ) + + "','" + escape( copyQuery.value( 3 ).toString() ) + + "','" + escape( copyQuery.value( 4 ).toString() ) + + "','" + escape( copyQuery.value( 5 ).toString() ) + + "','" + escape( copyQuery.value( 6 ).toString() ) + + "','" + escape( copyQuery.value( 7 ).toString() ) + + "')"; + database->exec( command ); + + emit progress(); + } + } + database->exec( "DROP TABLE recipes" ); + database->exec( "CREATE TABLE recipes (id INTEGER NOT NULL,title VARCHAR(200), yield_amount FLOAT, yield_amount_offset FLOAT, yield_type_id INTEGER DEFAULT '-1', instructions TEXT, photo BLOB, prep_time TIME, ctime TIMESTAMP, mtime TIMESTAMP, atime TIMESTAMP, PRIMARY KEY (id))" ); + copyQuery = database->exec( "SELECT id,title,yield_amount,yield_amount_offset,yield_type_id,instructions,photo,prep_time FROM recipes_copy" ); + if ( copyQuery.isActive() ) { + + TQString current_timestamp = TQDateTime::currentDateTime().toString(TQt::ISODate); + while ( copyQuery.next() ) { + command = "INSERT INTO recipes VALUES('" + + escape( copyQuery.value( 0 ).toString() ) + + "','" + escape( copyQuery.value( 1 ).toString() ) + + "','" + escape( copyQuery.value( 2 ).toString() ) + + "','" + escape( copyQuery.value( 3 ).toString() ) + + "','" + escape( copyQuery.value( 4 ).toString() ) + + "','" + escape( copyQuery.value( 5 ).toString() ) + + "','" + escape( copyQuery.value( 6 ).toString() ) + + "','" + escape( copyQuery.value( 7 ).toString() ) + + "','" + escape( current_timestamp ) //ctime + + "','" + escape( current_timestamp ) //mtime + + "','" + escape( current_timestamp ) //atime + + "')"; + database->exec( command ); + + emit progress(); + } + } + database->exec( "DROP TABLE recipes_copy" ); + + database->exec( "UPDATE db_info SET ver='0.84',generated_by='Krecipes SVN (20050913)';" ); + database->commit(); + } + + if ( tqRound(version*100) < 85 ) { + database->transaction(); + + TQSqlQuery query( "SELECT id,photo FROM recipes", database ); + + if ( query.isActive() ) { + while ( query.next() ) { + TQImage photo; + TQString photoString = query.value(1).toString(); + + // Decode the photo + uchar *photoArray = new uchar [ photoString.length() + 1 ]; + memcpy( photoArray, photoString.latin1(), photoString.length() * sizeof( char ) ); + sqlite_decode_binary( ( uchar* ) photoArray, ( uchar* ) photoArray ); + + photo.loadFromData( photoArray, photoString.length() ); + + // picture will now have a ready-to-use image + delete[] photoArray; + + TQByteArray ba; + TQBuffer buffer( ba ); + buffer.open( IO_WriteOnly ); + TQImageIO iio( &buffer, "JPEG" ); + iio.setImage( photo ); + iio.write(); + storePhoto( query.value(0).toInt(), ba ); + + emit progress(); + } + } + + + database->exec( "UPDATE db_info SET ver='0.85',generated_by='Krecipes SVN (20050926)';" ); + database->commit(); + } + + if ( tqRound(version*100) < 86 ) { + database->transaction(); + + database->exec( "CREATE index gidil_index ON ingredient_list(group_id)" ); + + TQSqlQuery query( "SELECT id,name FROM ingredient_groups ORDER BY name", database ); + + TQString last; + int lastID; + if ( query.isActive() ) { + while ( query.next() ) { + TQString name = query.value(1).toString(); + int id = query.value(0).toInt(); + if ( last == name ) { + TQString command = TQString("UPDATE ingredient_list SET group_id=%1 WHERE group_id=%2").arg(lastID).arg(id); + database->exec(command); + + command = TQString("DELETE FROM ingredient_groups WHERE id=%1").arg(id); + database->exec(command); + } + last = name; + lastID = id; + + emit progress(); + } + } + + database->exec( "UPDATE db_info SET ver='0.86',generated_by='Krecipes SVN (20050928)'" ); + if ( !database->commit() ) + kdDebug()<<"Update to 0.86 failed. Maybe you should try again."<<endl; + } + + if ( tqRound(version*100) < 87 ) { + //Load this default data so the user knows what rating criteria is + database->exec( TQString("INSERT INTO rating_criteria VALUES (1,'%1')").arg(i18n("Overall")) ); + database->exec( TQString("INSERT INTO rating_criteria VALUES (2,'%1')").arg(i18n("Taste") ) ); + database->exec( TQString("INSERT INTO rating_criteria VALUES (3,'%1')").arg(i18n("Appearance") ) ); + database->exec( TQString("INSERT INTO rating_criteria VALUES (4,'%1')").arg(i18n("Originality") ) ); + database->exec( TQString("INSERT INTO rating_criteria VALUES (5,'%1')").arg(i18n("Ease of Preparation") ) ); + + database->exec( "UPDATE db_info SET ver='0.87',generated_by='Krecipes SVN (20051014)'" ); + } + + if ( tqRound(version*100) < 90 ) { + database->exec("UPDATE db_info SET ver='0.9',generated_by='Krecipes 0.9'"); + } + + if ( tqRound(version*100) < 91 ) { + database->exec("CREATE index parent_id_index ON categories(parent_id)"); + database->exec("UPDATE db_info SET ver='0.91',generated_by='Krecipes SVN (20060526)'"); + } + + if ( tqRound(version*100) < 92 ) { + database->transaction(); + + //==================add a columns to 'units' to allow unit abbreviations + database->exec( "CREATE TABLE units_copy (id INTEGER NOT NULL, name VARCHAR(20), plural VARCHAR(20))" ); + TQSqlQuery copyQuery = database->exec( "SELECT id,name,plural FROM units" ); + if ( copyQuery.isActive() ) { + + while ( copyQuery.next() ) { + command = "INSERT INTO units_copy VALUES('" + + escape( copyQuery.value( 0 ).toString() ) //id + + "','" + escape( copyQuery.value( 1 ).toString() ) //name + + "','" + escape( copyQuery.value( 2 ).toString() ) //plural + + "')"; + database->exec( command ); + + emit progress(); + } + } + database->exec( "DROP TABLE units" ); + database->exec( "CREATE TABLE units (id INTEGER NOT NULL, name VARCHAR(20), name_abbrev VARCHAR(20), plural VARCHAR(20), plural_abbrev VARCHAR(20), PRIMARY KEY (id))" ); + copyQuery = database->exec( "SELECT id,name,plural FROM units_copy" ); + if ( copyQuery.isActive() ) { + while ( copyQuery.next() ) { + command = "INSERT INTO units VALUES('" + + escape( copyQuery.value( 0 ).toString() ) //id + + "','" + escape( copyQuery.value( 1 ).toString() ) //name + + "',NULL" //name_abbrev + + ",'" + escape( copyQuery.value( 2 ).toString() ) //plural + + "',NULL" //plural_abbrev + + ")"; + database->exec( command ); + + emit progress(); + } + } + database->exec( "DROP TABLE units_copy" ); + + database->exec("UPDATE db_info SET ver='0.92',generated_by='Krecipes SVN (20060609)'"); + if ( !database->commit() ) + kdDebug()<<"Update to 0.92 failed. Maybe you should try again."<<endl; + } + + if ( tqRound(version*100) < 93 ) { + database->transaction(); + + addColumn("CREATE TABLE %1 (id INTEGER NOT NULL, recipe_id INTEGER, ingredient_id INTEGER, amount FLOAT, amount_offset FLOAT, unit_id INTEGER, order_index INTEGER, group_id INTEGER, %2 PRIMARY KEY(id) )","substitute_for INTEGER","NULL","ingredient_list",8); + + database->exec( "CREATE index ridil_index ON ingredient_list(recipe_id)" ); + database->exec( "CREATE index iidil_index ON ingredient_list(ingredient_id)" ); + database->exec( "CREATE index gidil_index ON ingredient_list(group_id)" ); + + database->exec( "UPDATE db_info SET ver='0.93',generated_by='Krecipes SVN (20060616)';" ); + database->commit(); + } + + if ( tqRound(version*100) < 94 ) { + database->transaction(); + + //==================add a column to 'units' to allow specifying a type + database->exec( "CREATE TABLE units_copy (id INTEGER NOT NULL, name VARCHAR(20), name_abbrev VARCHAR(20), plural VARCHAR(20), plural_abbrev VARCHAR(20))" ); + TQSqlQuery copyQuery = database->exec( "SELECT id,name,name_abbrev,plural,plural_abbrev FROM units" ); + if ( copyQuery.isActive() ) { + + while ( copyQuery.next() ) { + TQString name_abbrev = escape( copyQuery.value( 2 ).toString() ); + if ( name_abbrev.isEmpty() ) + name_abbrev = "NULL"; + else { + name_abbrev.prepend("'"); + name_abbrev.append("'"); + } + TQString plural_abbrev = escape( copyQuery.value( 4 ).toString() ); + if ( plural_abbrev.isEmpty() ) + plural_abbrev = "NULL"; + else { + plural_abbrev.prepend("'"); + plural_abbrev.append("'"); + } + + command = "INSERT INTO units_copy VALUES('" + + escape( copyQuery.value( 0 ).toString() ) //id + + "','" + escape( copyQuery.value( 1 ).toString() ) //name + + "'," + name_abbrev //name_abbrev + + ",'" + escape( copyQuery.value( 3 ).toString() ) //plural + + "'," + plural_abbrev //plural_abbrev + + ")"; + database->exec( command ); + + emit progress(); + } + } + database->exec( "DROP TABLE units" ); + database->exec( "CREATE TABLE units (id INTEGER NOT NULL, name VARCHAR(20), name_abbrev VARCHAR(20), plural VARCHAR(20), plural_abbrev VARCHAR(20), type INTEGER NOT NULL, PRIMARY KEY (id))" ); + copyQuery = database->exec( "SELECT id,name,name_abbrev,plural,plural_abbrev FROM units_copy" ); + if ( copyQuery.isActive() ) { + + while ( copyQuery.next() ) { + TQString name_abbrev = escape( copyQuery.value( 2 ).toString() ); + if ( name_abbrev.isEmpty() ) + name_abbrev = "NULL"; + else { + name_abbrev.prepend("'"); + name_abbrev.append("'"); + } + TQString plural_abbrev = escape( copyQuery.value( 4 ).toString() ); + if ( plural_abbrev.isEmpty() ) + plural_abbrev = "NULL"; + else { + plural_abbrev.prepend("'"); + plural_abbrev.append("'"); + } + + command = "INSERT INTO units VALUES('" + + escape( copyQuery.value( 0 ).toString() ) //id + + "','" + escape( copyQuery.value( 1 ).toString() ) //name + + "'," + name_abbrev //name_abbrev + + ",'" + escape( copyQuery.value( 3 ).toString() ) //plural + + "'," + plural_abbrev //plural_abbrev + + ",'0')"; + database->exec( command ); + + emit progress(); + } + } + database->exec( "DROP TABLE units_copy" ); + + + database->exec( "UPDATE db_info SET ver='0.94',generated_by='Krecipes SVN (20060712)';" ); + database->commit(); + } + + if ( tqRound(version*100) < 95 ) { + database->exec( "DROP TABLE ingredient_weights" ); + createTable( "ingredient_weights" ); + database->exec( "UPDATE db_info SET ver='0.95',generated_by='Krecipes SVN (20060726)'" ); + } +} + +void LiteRecipeDB::addColumn( const TQString &new_table_sql, const TQString &new_col_info, const TQString &default_value, const TQString &table_name, int col_index ) +{ + TQString command; + + command = TQString(new_table_sql).arg(table_name+"_copy").arg(TQString::null); + kdDebug()<<"calling: "<<command<<endl; + database->exec( command ); + + command = "SELECT * FROM "+table_name; + kdDebug()<<"calling: "<<command<<endl; + TQSqlQuery copyQuery = database->exec( command ); + if ( copyQuery.isActive() ) { + while ( copyQuery.next() ) { + TQStringList dataList; + for ( int i = 0 ;; ++i ) { + if ( copyQuery.value(i).isNull() ) + break; + + TQString data = copyQuery.value(i).toString(); + + dataList << "'"+escape(data)+"'"; + } + command = "INSERT INTO "+table_name+"_copy VALUES("+dataList.join(",")+");"; + kdDebug()<<"calling: "<<command<<endl; + database->exec( command ); + + emit progress(); + } + } + database->exec( "DROP TABLE "+table_name ); + database->exec( TQString(new_table_sql).arg(table_name).arg(new_col_info+",") ); + copyQuery = database->exec( "SELECT * FROM "+table_name+"_copy" ); + if ( copyQuery.isActive() ) { + while ( copyQuery.next() ) { + TQStringList dataList; + for ( int i = 0 ;; ++i ) { + if ( i == col_index ) + dataList << default_value; + + if ( copyQuery.value(i).isNull() ) + break; + + TQString data = copyQuery.value(i).toString(); + + dataList << "'"+escape(data)+"'"; + } + command = "INSERT INTO "+table_name+" VALUES(" +dataList.join(",")+")"; + kdDebug()<<"calling: "<<command<<endl; + database->exec( command ); + + emit progress(); + } + } + database->exec( "DROP TABLE "+table_name+"_copy" ); +} + +TQString LiteRecipeDB::escapeAndEncode( const TQString &s ) const +{ + TQString s_escaped; + + // Escape + s_escaped = escape( TQString::fromLatin1(s.utf8()) ); + + // Return encoded + return s_escaped.latin1(); // Note that the text has already been converted before escaping. +} + +/* +** Decode the string "in" into binary data and write it into "out". +** This routine reverses the encoding created by sqlite_encode_binary(). +** The output will always be a few bytes less than the input. The number +** of bytes of output is returned. If the input is not a well-formed +** encoding, -1 is returned. +** +** The "in" and "out" parameters may point to the same buffer in order +** to decode a string in place. +*/ +int sqlite_decode_binary( const unsigned char *in, unsigned char *out ) +{ + int i, c, e; + e = *( in++ ); + i = 0; + while ( ( c = *( in++ ) ) != 0 ) { + if ( c == 1 ) { + c = *( in++ ); + if ( c == 1 ) { + c = 0; + } + else if ( c == 2 ) { + c = 1; + } + else if ( c == 3 ) { + c = '\''; + } + else { + return -1; + } + } + out[ i++ ] = ( c + e ) & 0xff; + } + return i; +} + +TQString escape( const TQString &s ) +{ + TQString s_escaped = s; + + if ( !s_escaped.isEmpty() ) { //###: sqlite_mprintf() seems to fill an empty string with garbage + // Escape using SQLite's function +#if HAVE_SQLITE + char * escaped = sqlite_mprintf( "%q", s.latin1() ); // Escape the string(allocates memory) +#elif HAVE_SQLITE3 + char * escaped = sqlite3_mprintf( "%q", s.latin1() ); // Escape the string(allocates memory) +#endif + s_escaped = escaped; +#if HAVE_SQLITE + sqlite_freemem( escaped ); // free allocated memory +#elif HAVE_SQLITE3 + sqlite3_free( escaped ); // free allocated memory +#endif + } + + return ( s_escaped ); +} + +#include "literecipedb.moc" diff --git a/src/backends/SQLite/literecipedb.h b/src/backends/SQLite/literecipedb.h new file mode 100644 index 0000000..f108599 --- /dev/null +++ b/src/backends/SQLite/literecipedb.h @@ -0,0 +1,66 @@ +/*************************************************************************** +* Copyright (C) 2003-2004 by * +* Unai Garro ([email protected]) * +* Jason Kivlighn ([email protected]) * +* Cyril Bosselut ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef LITERECIPEDB_H +#define LITERECIPEDB_H + +#include "backends/qsqlrecipedb.h" + +#include <tqstring.h> + +#ifdef HAVE_CONFIG_H +#include "config.h" +#endif + +#ifdef HAVE_SQLITE3 +#define SQLITE_DRIVER "TQSQLITE3" +#elif HAVE_SQLITE +#define SQLITE_DRIVER "TQSQLITE" +#endif + +class LiteRecipeDB : public TQSqlRecipeDB +{ + + TQ_OBJECT + +private: + void createDB( void ); + +public: + LiteRecipeDB( const TQString &DBName = DEFAULT_DB_NAME ); + ~LiteRecipeDB( void ); + + virtual int lastInsertID(); + + virtual void createTable( const TQString &tableName ); + virtual void givePermissions(const TQString&, const TQString&, const TQString&, const TQString&){} //no permissions in this backend + +protected: + virtual TQString qsqlDriverPlugin() const + { + return SQLITE_DRIVER; + } + + virtual TQString escapeAndEncode( const TQString &s ) const; + +private: + virtual void portOldDatabases( float version ); + virtual TQStringList backupCommand() const; + virtual TQStringList restoreCommand() const; + + void addColumn( const TQString &new_table_sql, const TQString &new_col_info, const TQString &default_value, const TQString &table_name, int col_index ); +}; + + + + +#endif diff --git a/src/backends/progressinterface.cpp b/src/backends/progressinterface.cpp new file mode 100644 index 0000000..ea5ce08 --- /dev/null +++ b/src/backends/progressinterface.cpp @@ -0,0 +1,79 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "progressinterface.h" + +#include <tqapplication.h> + +#include <kprogress.h> + +#include "recipedb.h" + +ProgressInterface::ProgressInterface( TQWidget *parent ) : progress_dlg(0), database(0), m_rate(1), m_rate_at(0) +{ + slot_obj = new ProgressSlotObject(parent,this); +} + +ProgressInterface::~ProgressInterface() +{ + listenOn(0); + delete slot_obj; +} + +void ProgressInterface::progressBegin( int steps, const TQString &caption, const TQString &text, int rate ) +{ + m_rate = rate; + + progress_dlg = new KProgressDialog((TQWidget*)slot_obj->parent(),0,caption,text,true); + + if ( steps == 0 ) + progress_dlg->progressBar()->setPercentageVisible(false); + progress_dlg->progressBar()->setTotalSteps( steps ); +} + +void ProgressInterface::progressDone() +{ + delete progress_dlg; + progress_dlg = 0; + + m_rate_at = 0; +} + +void ProgressInterface::progress() +{ + if ( progress_dlg->wasCancelled() ) { + database->cancelOperation(); + } + else { + ++m_rate_at; + + if ( m_rate_at % m_rate == 0 ) { + progress_dlg->progressBar()->advance(1); + m_rate_at = 0; + } + tqApp->processEvents(); + } +} + +void ProgressInterface::listenOn( RecipeDB *db ) +{ + if ( database) + database->disconnect(slot_obj); + + if ( db ) { + slot_obj->connect( db, TQ_SIGNAL(progressBegin(int,const TQString&,const TQString&,int)), slot_obj, TQ_SLOT(progressBegin(int,const TQString&,const TQString&,int)) ); + slot_obj->connect( db, TQ_SIGNAL(progressDone()), slot_obj, TQ_SLOT(progressDone()) ); + slot_obj->connect( db, TQ_SIGNAL(progress()), slot_obj, TQ_SLOT(progress()) ); + } + + database = db; +} + +#include "progressinterface.moc" diff --git a/src/backends/progressinterface.h b/src/backends/progressinterface.h new file mode 100644 index 0000000..de9fd28 --- /dev/null +++ b/src/backends/progressinterface.h @@ -0,0 +1,70 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef PROGRESSINTERFACE_H +#define PROGRESSINTERFACE_H + +#include <tqobject.h> +#include <tqwidget.h> + +class KProgressDialog; + +class RecipeDB; +class ProgressSlotObject; + +/** This class is used to monitor events from the database that may take + * awhile. Before a potentially long operation, the database will + * call progressBegin(), progress() a number of times, and then progressDone(). + * + * This class may be subclassed to perform certain operations during long + * operations by implementing the three virtual functions. The default + * implementation displays a progress bar dialog. + */ +class ProgressInterface +{ +public: + ProgressInterface( TQWidget *parent ); + ~ProgressInterface(); + + void listenOn( RecipeDB* ); + +protected: + friend class ProgressSlotObject; + + virtual void progressBegin(int,const TQString &caption,const TQString &text,int rate); + virtual void progressDone(); + virtual void progress(); + +private: + ProgressSlotObject *slot_obj; + KProgressDialog *progress_dlg; + RecipeDB *database; + + int m_rate; + int m_rate_at; +}; + +class ProgressSlotObject : public TQObject +{ +TQ_OBJECT + +public: + ProgressSlotObject( TQWidget*parent, ProgressInterface *p ) : TQObject(parent), pInterface(p){} + +public slots: + void progressBegin(int i,const TQString &caption=TQString::null,const TQString &text=TQString::null,int rate=1){ pInterface->progressBegin(i,caption,text,rate); } + void progressDone(){ pInterface->progressDone(); } + void progress(){ pInterface->progress(); } + +private: + ProgressInterface *pInterface; +}; + +#endif diff --git a/src/backends/qsqlrecipedb.cpp b/src/backends/qsqlrecipedb.cpp new file mode 100644 index 0000000..7d36a31 --- /dev/null +++ b/src/backends/qsqlrecipedb.cpp @@ -0,0 +1,2774 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* Copyright (C) 2004-2006 Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include <stdlib.h> + +#include "qsqlrecipedb.h" +#include "datablocks/categorytree.h" +#include "datablocks/rating.h" +#include "datablocks/weight.h" + +#include "propertycalculator.h" + +#include <tqbuffer.h> +#include <tqtextcodec.h> +#include <tqvariant.h> + +#include <tdeapplication.h> +#include <kdebug.h> +#include <kstandarddirs.h> +#include <tdetempfile.h> +#include <tdelocale.h> +#include <tdemessagebox.h> +#include <kmdcodec.h> + +int TQSqlRecipeDB::m_refCount = 0; + +TQSqlRecipeDB::TQSqlRecipeDB( const TQString &host, const TQString &user, const TQString &pass, const TQString &name, int port ) : RecipeDB(), + connectionName("connection" + TQString::number( m_refCount+1 )) +{ + DBuser = user; + DBpass = pass; + DBhost = host; + DBname = name; + DBport = port; + + dbOK = false; //it isn't ok until we've connect()'ed + ++m_refCount; + + TQTextCodec::setCodecForCStrings(TQTextCodec::codecForName("Latin1")); //this is the default, but let's explicitly set this to be sure +} + +TQSqlRecipeDB::~TQSqlRecipeDB() +{ + if ( dbOK ) { + database->close(); + } + + TQSqlDatabase::removeDatabase( connectionName ); + --m_refCount; +} + +void TQSqlRecipeDB::connect( bool create_db, bool create_tables ) +{ + kdDebug() << i18n( "TQSqlRecipeDB: Opening Database..." ) << endl; + kdDebug() << "Parameters: \n\thost: " << DBhost << "\n\tuser: " << DBuser << "\n\ttable: " << DBname << endl; + + bool driver_found = false; + + if ( qsqlDriver() ) //we're using a built-in driver + driver_found = true; + else { + TQStringList drivers = TQSqlDatabase::drivers(); + for ( TQStringList::const_iterator it = drivers.begin(); it != drivers.end(); ++it ) { + if ( ( *it ) == qsqlDriverPlugin() ) { + driver_found = true; + break; + } + } + } + + if ( !driver_found ) { + dbErr = TQString( i18n( "The TQt database plug-in (%1) is not installed. This plug-in is required for using this database backend." ) ).arg( qsqlDriverPlugin() ); + return ; + } + + //we need to have a unique connection name for each TQSqlRecipeDB class as multiple db's may be open at once (db to db transfer) + if ( qsqlDriver() ) + database = TQSqlDatabase::addDatabase( qsqlDriver(), connectionName ); + else if ( !qsqlDriverPlugin().isEmpty() ) + database = TQSqlDatabase::addDatabase( qsqlDriverPlugin(), connectionName ); + else + kdDebug()<<"Fatal internal error! Backend incorrectly written!"<<endl; + + database->setDatabaseName( DBname ); + if ( !( DBuser.isNull() ) ) + database->setUserName( DBuser ); + if ( !( DBpass.isNull() ) ) + database->setPassword( DBpass ); + database->setHostName( DBhost ); + if ( DBport > 0 ) + database->setPort(DBport); + + kdDebug() << i18n( "Parameters set. Calling db->open()" ) << endl; + + if ( !database->open() ) { + //Try to create the database + if ( create_db ) { + kdDebug() << i18n( "Failing to open database. Trying to create it" ) << endl; + createDB(); + } + else { + // Handle the error (passively) + dbErr = TQString( i18n( "Krecipes could not open the database using the driver '%2' (with username: \"%1\"). You may not have the necessary permissions, or the server may be down." ) ).arg( DBuser ).arg( qsqlDriverPlugin() ); + } + + //Now Reopen the Database and signal & exit if it fails + if ( !database->open() ) { + TQString error = i18n( "Database message: %1" ).arg( database->lastError().databaseText() ); + kdDebug() << i18n( "Failing to open database. Exiting\n" ).latin1(); + + // Handle the error (passively) + dbErr = TQString( i18n( "Krecipes could not open the database using the driver '%2' (with username: \"%1\"). You may not have the necessary permissions, or the server may be down." ) ).arg( DBuser ).arg( qsqlDriverPlugin() ); + return ; + } + } + + if ( int( tqRound( databaseVersion() * 1e5 ) ) > int( tqRound( latestDBVersion() * 1e5 ) ) ) { //correct for float's imprecision + dbErr = i18n( "This database was created with a newer version of Krecipes and cannot be opened." ); + return ; + } + + // Check integrity of the database (tables). If not possible, exit + // Because checkIntegrity() will create tables if they don't exist, + // we don't want to run this when creating the database. We would be + // logged in as another user (usually the superuser and not have ownership of the tables + if ( create_tables && !checkIntegrity() ) { + dbErr = i18n( "Failed to fix database structure.\nIf you are using SQLite, this is often caused by using an SQLite 2 database with SQLite 3 installed. If this is the case, make sure both SQLite 2 and 3 are installed, and then run 'krecipes --convert-sqlite3' to update your database to the new structure." ); + return; + } + + // Database was opened correctly + m_query = TQSqlQuery( TQString::null, database ); + m_query.setForwardOnly(true); + dbOK = true; +} + +void TQSqlRecipeDB::execSQL( const TQString &command ) +{ + database->exec( command ); +} + +void TQSqlRecipeDB::loadRecipes( RecipeList *rlist, int items, TQValueList<int> ids ) +{ + // Empty the recipe first + rlist->empty(); + + TQMap <int, RecipeList::Iterator> recipeIterators; // Stores the iterator of each recipe in the list; + + TQString command; + + TQString current_timestamp = TQDateTime::currentDateTime().toString(TQt::ISODate); + + TQStringList ids_str; + for ( TQValueList<int>::const_iterator it = ids.begin(); it != ids.end(); ++it ) { + TQString number_str = TQString::number(*it); + ids_str << number_str; + + if ( !(items & RecipeDB::Noatime) ) + database->exec( "UPDATE recipes SET ctime=ctime,mtime=mtime,atime='"+current_timestamp+"' WHERE id="+number_str ); + } + + // Read title, author, yield, and instructions as specified + command = "SELECT id"; + if ( items & RecipeDB::Title ) command += ",title"; + if ( items & RecipeDB::Instructions ) command += ",instructions"; + if ( items & RecipeDB::PrepTime ) command += ",prep_time"; + if ( items & RecipeDB::Yield ) command += ",yield_amount,yield_amount_offset,yield_type_id"; + command += " FROM recipes"+(ids_str.count()!=0?" WHERE id IN ("+ids_str.join(",")+")":""); + + TQSqlQuery recipeQuery(command,database); + if ( recipeQuery.isActive() ) { + while ( recipeQuery.next() ) { + int row_at = 0; + + Recipe recipe; + recipe.recipeID = recipeQuery.value( row_at ).toInt(); ++row_at; + + if ( items & RecipeDB::Title ) { + recipe.title = unescapeAndDecode( recipeQuery.value( row_at ).toCString() ); ++row_at; + } + + if ( items & RecipeDB::Instructions ) { + recipe.instructions = unescapeAndDecode( recipeQuery.value( row_at ).toCString() ); ++row_at; + } + + if ( items & RecipeDB::PrepTime ) { + recipe.prepTime = recipeQuery.value( row_at ).toTime(); ++row_at; + } + + if ( items & RecipeDB::Yield ) { + recipe.yield.amount = recipeQuery.value( row_at ).toDouble(); ++row_at; + recipe.yield.amount_offset = recipeQuery.value( row_at ).toDouble(); ++row_at; + recipe.yield.type_id = recipeQuery.value( row_at ).toInt(); ++row_at; + if ( recipe.yield.type_id != -1 ) { + TQString y_command = TQString("SELECT name FROM yield_types WHERE id=%1;").arg(recipe.yield.type_id); + TQSqlQuery yield_query(y_command,database); + if ( yield_query.isActive() && yield_query.first() ) + recipe.yield.type = unescapeAndDecode(yield_query.value( 0 ).toCString()); + else + kdDebug()<<yield_query.lastError().databaseText()<<endl; + } + } + + if ( items & RecipeDB::Meta ) + loadRecipeMetadata(&recipe); + + recipeIterators[ recipe.recipeID ] = rlist->append( recipe ); + } + } + + // Read the ingredients + if ( items & RecipeDB::Ingredients ) { + for ( RecipeList::iterator recipe_it = rlist->begin(); recipe_it != rlist->end(); ++recipe_it ) { + if ( items & RecipeDB::NamesOnly ) { + if ( items & IngredientAmounts ) + command = TQString( "SELECT il.ingredient_id,i.name,il.substitute_for,il.amount,il.amount_offset,u.id,u.type FROM ingredients i, ingredient_list il, units u WHERE il.recipe_id=%1 AND i.id=il.ingredient_id AND u.id=il.unit_id ORDER BY il.order_index" ).arg( (*recipe_it).recipeID ); + else + command = TQString( "SELECT il.ingredient_id,i.name,il.substitute_for FROM ingredients i, ingredient_list il WHERE il.recipe_id=%1 AND i.id=il.ingredient_id" ).arg( (*recipe_it).recipeID ); + } + else + command = TQString( "SELECT il.ingredient_id,i.name,il.substitute_for,il.amount,il.amount_offset,u.id,u.name,u.plural,u.name_abbrev,u.plural_abbrev,u.type,il.group_id,il.id FROM ingredients i, ingredient_list il, units u WHERE il.recipe_id=%1 AND i.id=il.ingredient_id AND u.id=il.unit_id ORDER BY il.order_index" ).arg( (*recipe_it).recipeID ); + + TQSqlQuery ingredientQuery( command, database ); + if ( ingredientQuery.isActive() ) { + RecipeList::Iterator it = recipeIterators[ (*recipe_it).recipeID ]; + while ( ingredientQuery.next() ) { + Ingredient ing; + ing.ingredientID = ingredientQuery.value( 0 ).toInt(); + ing.name = unescapeAndDecode( ingredientQuery.value( 1 ).toCString() ); + + if ( items & RecipeDB::NamesOnly ) { + if ( items & IngredientAmounts ) { + ing.amount = ingredientQuery.value( 3 ).toDouble(); + ing.amount_offset = ingredientQuery.value( 4 ).toDouble(); + ing.units.id = ingredientQuery.value( 5 ).toInt(); + ing.units.type = (Unit::Type)ingredientQuery.value( 6 ).toInt(); + } + } + else { + ing.amount = ingredientQuery.value( 3 ).toDouble(); + ing.amount_offset = ingredientQuery.value( 4 ).toDouble(); + ing.units.id = ingredientQuery.value( 5 ).toInt(); + ing.units.name = unescapeAndDecode( ingredientQuery.value( 6 ).toCString() ); + ing.units.plural = unescapeAndDecode( ingredientQuery.value( 7 ).toCString() ); + ing.units.name_abbrev = unescapeAndDecode( ingredientQuery.value( 8 ).toCString() ); + ing.units.plural_abbrev = unescapeAndDecode( ingredientQuery.value( 9 ).toCString() ); + ing.units.type = (Unit::Type)ingredientQuery.value( 10 ).toInt(); + + //if we don't have both name and plural, use what we have as both + if ( ing.units.name.isEmpty() ) + ing.units.name = ing.units.plural; + else if ( ing.units.plural.isEmpty() ) + ing.units.plural = ing.units.name; + + ing.groupID = ingredientQuery.value( 11 ).toInt(); + if ( ing.groupID != -1 ) { + TQSqlQuery toLoad( TQString( "SELECT name FROM ingredient_groups WHERE id=%1" ).arg( ing.groupID ), database ); + if ( toLoad.isActive() && toLoad.first() ) + ing.group = unescapeAndDecode( toLoad.value( 0 ).toCString() ); + } + + command = TQString("SELECT pl.prep_method_id,p.name FROM prep_methods p, prep_method_list pl WHERE pl.ingredient_list_id=%1 AND p.id=pl.prep_method_id ORDER BY pl.order_index;").arg(ingredientQuery.value( 12 ).toInt()); + TQSqlQuery ingPrepMethodsQuery( command, database ); + if ( ingPrepMethodsQuery.isActive() ) { + while ( ingPrepMethodsQuery.next() ) { + ing.prepMethodList.append( Element( unescapeAndDecode(ingPrepMethodsQuery.value(1).toCString()),ingPrepMethodsQuery.value(0).toInt()) ); + } + } + } + + if ( ingredientQuery.value( 2 ).toInt() > 0 ) { + //given the ordering, we can assume substitute_for is the id of the last + //ingredient in the list + //int substitute_for = ingredientQuery.value( 2 ).toInt(); + (*it).ingList.last().substitutes.append( ing ); + } + else + (*it).ingList.append( ing ); + } + } + } + } + + //Load the Image + if ( items & RecipeDB::Photo ) { + for ( RecipeList::iterator recipe_it = rlist->begin(); recipe_it != rlist->end(); ++recipe_it ) { + RecipeList::iterator it = recipeIterators[ (*recipe_it).recipeID ]; + loadPhoto( (*it).recipeID, (*it).photo ); + } + } + + //Load the category list + if ( items & RecipeDB::Categories ) { + for ( RecipeList::iterator recipe_it = rlist->begin(); recipe_it != rlist->end(); ++recipe_it ) { + RecipeList::iterator it = recipeIterators[ (*recipe_it).recipeID ]; + + command = TQString( "SELECT cl.category_id,c.name FROM category_list cl, categories c WHERE recipe_id=%1 AND cl.category_id=c.id;" ).arg( (*it).recipeID ); + + m_query.exec( command ); + if ( m_query.isActive() ) { + while ( m_query.next() ) { + Element el; + el.id = m_query.value( 0 ).toInt(); + el.name = unescapeAndDecode( m_query.value( 1 ).toCString() ); + (*it).categoryList.append( el ); + } + } + } + } + + //Load the author list + if ( items & RecipeDB::Authors ) { + for ( RecipeList::iterator recipe_it = rlist->begin(); recipe_it != rlist->end(); ++recipe_it ) { + RecipeList::iterator it = recipeIterators[ (*recipe_it).recipeID ]; + + command = TQString( "SELECT al.author_id,a.name FROM author_list al, authors a WHERE recipe_id=%1 AND al.author_id=a.id;" ).arg( (*it).recipeID ); + + m_query.exec( command ); + if ( m_query.isActive() ) { + while ( m_query.next() ) { + Element el; + el.id = m_query.value( 0 ).toInt(); + el.name = unescapeAndDecode( m_query.value( 1 ).toCString() ); + (*it).authorList.append( el ); + } + } + } + } + + //Load the ratings + if ( items & RecipeDB::Ratings ) { + for ( RecipeList::iterator recipe_it = rlist->begin(); recipe_it != rlist->end(); ++recipe_it ) { + RecipeList::iterator it = recipeIterators[ (*recipe_it).recipeID ]; + + command = TQString( "SELECT id,comment,rater FROM ratings WHERE recipe_id=%1 ORDER BY created DESC" ).arg( (*it).recipeID ); + TQSqlQuery query( command, database ); + if ( query.isActive() ) { + while ( query.next() ) { + Rating r; + r.id = query.value( 0 ).toInt(); + r.comment = unescapeAndDecode( query.value( 1 ).toCString() ); + r.rater = unescapeAndDecode( query.value( 2 ).toCString() ); + + command = TQString( "SELECT rc.id,rc.name,rl.stars FROM rating_criteria rc, rating_criterion_list rl WHERE rating_id=%1 AND rl.rating_criterion_id=rc.id" ).arg(r.id); + TQSqlQuery criterionQuery( command, database ); + if ( criterionQuery.isActive() ) { + while ( criterionQuery.next() ) { + RatingCriteria rc; + rc.id = criterionQuery.value( 0 ).toInt(); + rc.name = unescapeAndDecode( criterionQuery.value( 1 ).toCString() ); + rc.stars = criterionQuery.value( 2 ).toDouble(); + r.append( rc ); + } + } + + (*it).ratingList.append( r ); + } + } + } + } + + if ( items & RecipeDB::Properties ) { + for ( RecipeList::iterator recipe_it = rlist->begin(); recipe_it != rlist->end(); ++recipe_it ) { + RecipeList::iterator it = recipeIterators[ (*recipe_it).recipeID ]; + calculateProperties( *it, this ); + } + } +} + +void TQSqlRecipeDB::loadIngredientGroups( ElementList *list ) +{ + list->clear(); + + TQString command = "SELECT id,name FROM ingredient_groups ORDER BY name;"; + m_query.exec( command ); + + if ( m_query.isActive() ) { + while ( m_query.next() ) { + Element group; + group.id = m_query.value( 0 ).toInt(); + group.name = unescapeAndDecode( m_query.value( 1 ).toCString() ); + list->append( group ); + } + } +} + +void TQSqlRecipeDB::loadIngredients( ElementList *list, int limit, int offset ) +{ + list->clear(); + + TQString command = "SELECT id,name FROM ingredients ORDER BY name" + +((limit==-1)?"":" LIMIT "+TQString::number(limit)+" OFFSET "+TQString::number(offset)); + m_query.exec( command ); + + if ( m_query.isActive() ) { + while ( m_query.next() ) { + Element ing; + ing.id = m_query.value( 0 ).toInt(); + ing.name = unescapeAndDecode( m_query.value( 1 ).toCString() ); + list->append( ing ); + } + } +} + +void TQSqlRecipeDB::loadPrepMethods( ElementList *list, int limit, int offset ) +{ + list->clear(); + + TQString command = "SELECT id,name FROM prep_methods ORDER BY name" + +((limit==-1)?"":" LIMIT "+TQString::number(limit)+" OFFSET "+TQString::number(offset)); + m_query.exec( command ); + + if ( m_query.isActive() ) { + while ( m_query.next() ) { + Element prep_method; + prep_method.id = m_query.value( 0 ).toInt(); + prep_method.name = unescapeAndDecode( m_query.value( 1 ).toCString() ); + list->append( prep_method ); + } + } +} + +void TQSqlRecipeDB::loadYieldTypes( ElementList *list, int limit, int offset ) +{ + list->clear(); + + TQString command = "SELECT id,name FROM yield_types ORDER BY name" + +((limit==-1)?"":" LIMIT "+TQString::number(limit)+" OFFSET "+TQString::number(offset)); + m_query.exec( command ); + + if ( m_query.isActive() ) { + while ( m_query.next() ) { + Element el; + el.id = m_query.value( 0 ).toInt(); + el.name = unescapeAndDecode( m_query.value( 1 ).toCString() ); + list->append( el ); + } + } +} + +void TQSqlRecipeDB::createNewPrepMethod( const TQString &prepMethodName ) +{ + TQString command; + TQString real_name = prepMethodName.left( maxPrepMethodNameLength() ); + + command = TQString( "INSERT INTO prep_methods VALUES(%2,'%1');" ).arg( escapeAndEncode( real_name ) ).arg( getNextInsertIDStr( "prep_methods", "id" ) ); + TQSqlQuery prepMethodToCreate( command, database ); + + emit prepMethodCreated( Element( real_name, lastInsertID() ) ); +} + +void TQSqlRecipeDB::modPrepMethod( int prepMethodID, const TQString &newLabel ) +{ + TQString command; + + command = TQString( "UPDATE prep_methods SET name='%1' WHERE id=%2;" ).arg( escapeAndEncode( newLabel ) ).arg( prepMethodID ); + TQSqlQuery prepMethodToCreate( command, database ); + + emit prepMethodRemoved( prepMethodID ); + emit prepMethodCreated( Element( newLabel, prepMethodID ) ); +} + +void TQSqlRecipeDB::modProperty( int propertyID, const TQString &newLabel ) +{ + TQString command; + + command = TQString( "UPDATE ingredient_properties SET name='%1' WHERE id=%2;" ).arg( escapeAndEncode( newLabel ) ).arg( propertyID ); + TQSqlQuery createQuery( command, database ); + + emit propertyRemoved( propertyID ); + emit propertyCreated( propertyName( propertyID ) ); +} + +void TQSqlRecipeDB::loadPossibleUnits( int ingredientID, UnitList *list ) +{ + list->clear(); + + TQString command; + + command = TQString( "SELECT u.id,u.name,u.plural,u.name_abbrev,u.plural_abbrev,u.type FROM unit_list ul, units u WHERE ul.ingredient_id=%1 AND ul.unit_id=u.id;" ).arg( ingredientID ); + + TQSqlQuery unitToLoad( command, database ); + + if ( unitToLoad.isActive() ) { + while ( unitToLoad.next() ) { + Unit unit; + unit.id = unitToLoad.value( 0 ).toInt(); + unit.name = unescapeAndDecode( unitToLoad.value( 1 ).toCString() ); + unit.plural = unescapeAndDecode( unitToLoad.value( 2 ).toCString() ); + unit.name_abbrev = unescapeAndDecode( unitToLoad.value( 3 ).toCString() ); + unit.plural_abbrev = unescapeAndDecode( unitToLoad.value( 4 ).toCString() ); + unit.type = (Unit::Type) unitToLoad.value( 5 ).toInt(); + + list->append( unit ); + } + } + + +} + +void TQSqlRecipeDB::storePhoto( int recipeID, const TQByteArray &data ) +{ + TQSqlQuery query( TQString::null, database ); + + query.prepare( "UPDATE recipes SET photo=?,ctime=ctime,atime=atime,mtime=mtime WHERE id=" + TQString::number( recipeID ) ); + query.addBindValue( KCodecs::base64Encode( data ) ); + query.exec(); +} + +void TQSqlRecipeDB::loadPhoto( int recipeID, TQPixmap &photo ) +{ + TQString command = TQString( "SELECT photo FROM recipes WHERE id=%1;" ).arg( recipeID ); + TQSqlQuery query( command, database ); + + if ( query.isActive() && query.first() ) { + TQCString decodedPic; + TQPixmap pix; + KCodecs::base64Decode( query.value( 0 ).toCString(), decodedPic ); + int len = decodedPic.size(); + + if ( len > 0 ) { + TQByteArray picData( len ); + memcpy( picData.data(), decodedPic.data(), len ); + + bool ok = pix.loadFromData( picData, "JPEG" ); + if ( ok ) + photo = pix; + } + } +} + +void TQSqlRecipeDB::loadRecipeMetadata( Recipe *recipe ) +{ + TQString command = "SELECT ctime,mtime,atime FROM recipes WHERE id="+TQString::number(recipe->recipeID); + + TQSqlQuery query( command, database ); + if ( query.isActive() && query.first() ) { + recipe->ctime = query.value(0).toDateTime(); + recipe->mtime = query.value(1).toDateTime(); + recipe->atime = query.value(2).toDateTime(); + } +} + +void TQSqlRecipeDB::saveRecipe( Recipe *recipe ) +{ + // Check if it's a new recipe or it exists (supossedly) already. + + bool newRecipe; + newRecipe = ( recipe->recipeID == -1 ); + // First check if the recipe ID is set, if so, update (not create) + // Be carefull, first check if the recipe hasn't been deleted while changing. + + TQSqlQuery recipeToSave( TQString::null, database ); + + TQString command; + + TQDateTime current_datetime = TQDateTime::currentDateTime(); + TQString current_timestamp = current_datetime.toString(TQt::ISODate); + if ( newRecipe ) { + command = TQString( "INSERT INTO recipes VALUES ("+getNextInsertIDStr("recipes","id")+",'%1',%2,'%3','%4','%5',NULL,'%6','%7','%8','%9');" ) // Id is autoincremented + .arg( escapeAndEncode( recipe->title ) ) + .arg( recipe->yield.amount ) + .arg( recipe->yield.amount_offset ) + .arg( recipe->yield.type_id ) + .arg( escapeAndEncode( recipe->instructions ) ) + .arg( recipe->prepTime.toString( "hh:mm:ss" ) ) + .arg( current_timestamp ) + .arg( current_timestamp ) + .arg( current_timestamp ); + recipe->mtime = recipe->ctime = recipe->atime = current_datetime; + } + else { + command = TQString( "UPDATE recipes SET title='%1',yield_amount='%2',yield_amount_offset='%3',yield_type_id='%4',instructions='%5',prep_time='%6',mtime='%8',ctime=ctime,atime=atime WHERE id=%7;" ) + .arg( escapeAndEncode( recipe->title ) ) + .arg( recipe->yield.amount ) + .arg( recipe->yield.amount_offset ) + .arg( recipe->yield.type_id ) + .arg( escapeAndEncode( recipe->instructions ) ) + .arg( recipe->prepTime.toString( "hh:mm:ss" ) ) + .arg( recipe->recipeID ) + .arg( current_timestamp ); + recipe->mtime = current_datetime; + } + recipeToSave.exec( command ); + + if ( !newRecipe ) { + // Clean up yield_types which have no recipe that they belong to + TQStringList ids; + command = TQString( "SELECT DISTINCT(yield_type_id) FROM recipes" ); + recipeToSave.exec( command ); + if ( recipeToSave.isActive() ) { + while ( recipeToSave.next() ) { + if ( recipeToSave.value( 0 ).toInt() != -1 ) + ids << TQString::number( recipeToSave.value( 0 ).toInt() ); + } + } + command = TQString( "DELETE FROM yield_types WHERE id NOT IN ( %1 )" ).arg( ( ids.count() == 0 ) ? "-1" : ids.join( "," ) ); + recipeToSave.exec( command ); + } + + // If it's a new recipe, identify the ID that was given to the recipe and store in the Recipe itself + int recipeID; + if ( newRecipe ) { + recipeID = lastInsertID(); + recipe->recipeID = recipeID; + } + recipeID = recipe->recipeID; + loadRecipeMetadata(recipe); + + // Let's begin storing the Image! + if ( !recipe->photo.isNull() ) { + TQByteArray ba; + TQBuffer buffer( ba ); + buffer.open( IO_WriteOnly ); + TQImageIO iio( &buffer, "JPEG" ); + iio.setImage( recipe->photo.convertToImage() ); + iio.write(); + + storePhoto( recipeID, ba ); + } + else { + recipeToSave.exec( "UPDATE recipes SET photo=NULL, mtime=mtime, ctime=ctime, atime=atime WHERE id=" + TQString::number( recipeID ) ); + } + + // Save the ingredient list (first delete if we are updating) + command = TQString( "SELECT id FROM ingredient_list WHERE recipe_id=%1" ).arg(recipeID); + recipeToSave.exec( command ); + if ( recipeToSave.isActive() ) { + while ( recipeToSave.next() ) { + command = TQString("DELETE FROM prep_method_list WHERE ingredient_list_id=%1") + .arg(recipeToSave.value(0).toInt()); + database->exec(command); + } + } + command = TQString( "DELETE FROM ingredient_list WHERE recipe_id=%1;" ) + .arg( recipeID ); + recipeToSave.exec( command ); + + int order_index = 0; + for ( IngredientList::const_iterator ing_it = recipe->ingList.begin(); ing_it != recipe->ingList.end(); ++ing_it ) { + order_index++; + TQString ing_list_id_str = getNextInsertIDStr("ingredient_list","id"); + command = TQString( "INSERT INTO ingredient_list VALUES (%1,%2,%3,%4,%5,%6,%7,%8,NULL);" ) + .arg( ing_list_id_str ) + .arg( recipeID ) + .arg( ( *ing_it ).ingredientID ) + .arg( ( *ing_it ).amount ) + .arg( ( *ing_it ).amount_offset ) + .arg( ( *ing_it ).units.id ) + .arg( order_index ) + .arg( ( *ing_it ).groupID ); + recipeToSave.exec( command ); + + int ing_list_id = lastInsertID(); + int prep_order_index = 0; + for ( ElementList::const_iterator prep_it = (*ing_it).prepMethodList.begin(); prep_it != (*ing_it).prepMethodList.end(); ++prep_it ) { + prep_order_index++; + command = TQString( "INSERT INTO prep_method_list VALUES (%1,%2,%3);" ) + .arg( ing_list_id ) + .arg( ( *prep_it ).id ) + .arg( prep_order_index ); + recipeToSave.exec( command ); + } + + for ( TQValueList<IngredientData>::const_iterator sub_it = (*ing_it).substitutes.begin(); sub_it != (*ing_it).substitutes.end(); ++sub_it ) { + order_index++; + TQString ing_list_id_str = getNextInsertIDStr("ingredient_list","id"); + command = TQString( "INSERT INTO ingredient_list VALUES (%1,%2,%3,%4,%5,%6,%7,%8,%9);" ) + .arg( ing_list_id_str ) + .arg( recipeID ) + .arg( ( *sub_it ).ingredientID ) + .arg( ( *sub_it ).amount ) + .arg( ( *sub_it ).amount_offset ) + .arg( ( *sub_it ).units.id ) + .arg( order_index ) + .arg( ( *sub_it ).groupID ) + .arg( (*ing_it).ingredientID ); + recipeToSave.exec( command ); + + int ing_list_id = lastInsertID(); + int prep_order_index = 0; + for ( ElementList::const_iterator prep_it = (*sub_it).prepMethodList.begin(); prep_it != (*sub_it).prepMethodList.end(); ++prep_it ) { + prep_order_index++; + command = TQString( "INSERT INTO prep_method_list VALUES (%1,%2,%3);" ) + .arg( ing_list_id ) + .arg( ( *prep_it ).id ) + .arg( prep_order_index ); + recipeToSave.exec( command ); + } + } + } + + // Save the category list for the recipe (first delete, in case we are updating) + command = TQString( "DELETE FROM category_list WHERE recipe_id=%1;" ) + .arg( recipeID ); + recipeToSave.exec( command ); + + ElementList::const_iterator cat_it = recipe->categoryList.end(); // Start from last, mysql seems to work in lifo format... so it's read first the latest inserted one (newest) + --cat_it; + for ( unsigned int i = 0; i < recipe->categoryList.count(); i++ ) { + command = TQString( "INSERT INTO category_list VALUES (%1,%2);" ) + .arg( recipeID ) + .arg( ( *cat_it ).id ); + recipeToSave.exec( command ); + + --cat_it; + } + + //Add the default category -1 to ease and speed up searches + + command = TQString( "INSERT INTO category_list VALUES (%1,-1);" ) + .arg( recipeID ); + recipeToSave.exec( command ); + + + // Save the author list for the recipe (first delete, in case we are updating) + command = TQString( "DELETE FROM author_list WHERE recipe_id=%1;" ) + .arg( recipeID ); + recipeToSave.exec( command ); + + ElementList::const_iterator author_it = recipe->authorList.end(); // Start from last, mysql seems to work in lifo format... so it's read first the latest inserted one (newest) + --author_it; + for ( unsigned int i = 0; i < recipe->authorList.count(); i++ ) { + command = TQString( "INSERT INTO author_list VALUES (%1,%2);" ) + .arg( recipeID ) + .arg( ( *author_it ).id ); + recipeToSave.exec( command ); + + --author_it; + } + + // Save the ratings (first delete criterion list if we are updating) + command = TQString( "SELECT id FROM ratings WHERE recipe_id=%1" ).arg(recipeID); + recipeToSave.exec( command ); + if ( recipeToSave.isActive() ) { + while ( recipeToSave.next() ) { + + command = TQString("DELETE FROM rating_criterion_list WHERE rating_id=%1") + .arg(recipeToSave.value(0).toInt()); + database->exec(command); + } + } + + TQStringList ids; + + for ( RatingList::iterator rating_it = recipe->ratingList.begin(); rating_it != recipe->ratingList.end(); ++rating_it ) { + //double average = (*rating_it).average(); + if ( (*rating_it).id == -1 ) //new rating + command ="INSERT INTO ratings VALUES("+TQString(getNextInsertIDStr("ratings","id"))+","+TQString::number(recipeID)+",'"+TQString(escapeAndEncode((*rating_it).comment))+"','"+TQString(escapeAndEncode((*rating_it).rater))+/*"','"+TQString::number(average)+*/"','"+current_timestamp+"')"; + else //existing rating + command = "UPDATE ratings SET " + "comment='"+TQString(escapeAndEncode((*rating_it).comment))+"'," + "rater='"+TQString(escapeAndEncode((*rating_it).rater))+"'," + "created=created " + "WHERE id="+TQString::number((*rating_it).id); + + recipeToSave.exec( command ); + + if ( (*rating_it).id == -1 ) + (*rating_it).id = lastInsertID(); + + for ( TQValueList<RatingCriteria>::const_iterator rc_it = (*rating_it).ratingCriteriaList.begin(); rc_it != (*rating_it).ratingCriteriaList.end(); ++rc_it ) { + command = TQString( "INSERT INTO rating_criterion_list VALUES("+TQString::number((*rating_it).id)+","+TQString::number((*rc_it).id)+","+TQString::number((*rc_it).stars)+")" ); + recipeToSave.exec( command ); + } + + ids << TQString::number((*rating_it).id); + } + + // only delete those ratings that don't exist anymore + command = TQString( "DELETE FROM ratings WHERE recipe_id=%1 AND id NOT IN( %2 )" ) + .arg( recipeID ).arg( ids.join(",") ); + recipeToSave.exec( command ); + + if ( newRecipe ) + emit recipeCreated( Element( recipe->title.left( maxRecipeTitleLength() ), recipeID ), recipe->categoryList ); + else + emit recipeModified( Element( recipe->title.left( maxRecipeTitleLength() ), recipeID ), recipe->categoryList ); +} + +void TQSqlRecipeDB::loadRecipeList( ElementList *list, int categoryID, bool recursive ) +{ + TQString command; + + if ( categoryID == -1 ) // load just the list + command = "SELECT id,title FROM recipes;"; + else // load the list of those in the specified category + command = TQString( "SELECT r.id,r.title FROM recipes r,category_list cl WHERE r.id=cl.recipe_id AND cl.category_id=%1 ORDER BY r.title" ).arg( categoryID ); + + if ( recursive ) { + TQSqlQuery subcategories( TQString("SELECT id FROM categories WHERE parent_id='%1'").arg(categoryID), database ); + if ( subcategories.isActive() ) { + while ( subcategories.next() ) { + loadRecipeList(list,subcategories.value( 0 ).toInt(),true); + } + } + } + + TQSqlQuery recipeToLoad( command, database ); + + if ( recipeToLoad.isActive() ) { + while ( recipeToLoad.next() ) { + Element recipe; + recipe.id = recipeToLoad.value( 0 ).toInt(); + recipe.name = unescapeAndDecode( recipeToLoad.value( 1 ).toCString() ); + list->append( recipe ); + } + } +} + + +void TQSqlRecipeDB::loadUncategorizedRecipes( ElementList *list ) +{ + list->clear(); + + TQString command = "SELECT r.id,r.title FROM recipes r,category_list cl WHERE r.id=cl.recipe_id GROUP BY id HAVING COUNT(*)=1 ORDER BY r.title DESC"; + m_query.exec( command ); + if ( m_query.isActive() ) { + while ( m_query.next() ) { + Element recipe; + recipe.id = m_query.value( 0 ).toInt(); + recipe.name = unescapeAndDecode( m_query.value( 1 ).toCString() ); + list->append( recipe ); + } + } +} + + + +void TQSqlRecipeDB::removeRecipe( int id ) +{ + emit recipeRemoved( id ); + + TQString command; + + command = TQString( "DELETE FROM recipes WHERE id=%1;" ).arg( id ); + TQSqlQuery recipeToRemove( command, database ); + command = TQString( "DELETE FROM ingredient_list WHERE recipe_id=%1;" ).arg( id ); + recipeToRemove.exec( command ); + command = TQString( "DELETE FROM category_list WHERE recipe_id=%1;" ).arg( id ); + recipeToRemove.exec( command ); + + // Clean up ingredient_groups which have no recipe that they belong to + // MySQL doesn't support subqueries until 4.1, so we'll do this the long way + // (Easy way: DELETE FROM ingredient_groups WHERE id NOT IN ( SELECT DISTINCT(group_id) FROM ingredient_list );) + TQStringList ids; + command = TQString( "SELECT DISTINCT(group_id) FROM ingredient_list;" ); + recipeToRemove.exec( command ); + if ( recipeToRemove.isActive() ) { + while ( recipeToRemove.next() ) { + if ( recipeToRemove.value( 0 ).toInt() != -1 ) + ids << TQString::number( recipeToRemove.value( 0 ).toInt() ); + } + } + command = TQString( "DELETE FROM ingredient_groups WHERE id NOT IN ( %1 );" ).arg( ( ids.count() == 0 ) ? "-1" : ids.join( "," ) ); + recipeToRemove.exec( command ); + + // Clean up yield_types which have no recipe that they belong to + ids.clear(); + command = TQString( "SELECT DISTINCT(yield_type_id) FROM recipes" ); + recipeToRemove.exec( command ); + if ( recipeToRemove.isActive() ) { + while ( recipeToRemove.next() ) { + if ( recipeToRemove.value( 0 ).toInt() != -1 ) + ids << TQString::number( recipeToRemove.value( 0 ).toInt() ); + } + } + command = TQString( "DELETE FROM yield_types WHERE id NOT IN ( %1 );" ).arg( ( ids.count() == 0 ) ? "-1" : ids.join( "," ) ); + recipeToRemove.exec( command ); +} + +void TQSqlRecipeDB::removeRecipeFromCategory( int recipeID, int categoryID ) +{ + TQString command; + command = TQString( "DELETE FROM category_list WHERE recipe_id=%1 AND category_id=%2;" ).arg( recipeID ).arg( categoryID ); + TQSqlQuery recipeToRemove( command, database ); + + emit recipeRemoved( recipeID, categoryID ); +} + +void TQSqlRecipeDB::categorizeRecipe( int recipeID, const ElementList &categoryList ) +{ + TQString command; + + //emit recipeRemoved( recipeID, -1 ); + + for ( ElementList::const_iterator it = categoryList.begin(); it != categoryList.end(); ++it ) { + command = TQString( "INSERT INTO category_list VALUES(%1,%2)" ).arg( recipeID ).arg( (*it).id ); + database->exec( command ); + } + + emit recipeModified( Element(recipeTitle(recipeID),recipeID), categoryList ); +} + +void TQSqlRecipeDB::createNewIngGroup( const TQString &name ) +{ + TQString command; + TQString real_name = name.left( maxIngGroupNameLength() ); + + command = TQString( "INSERT INTO ingredient_groups VALUES(%2,'%1');" ).arg( escapeAndEncode( real_name ) ).arg( getNextInsertIDStr( "ingredient_groups", "id" ) ); + TQSqlQuery query( command, database ); + + emit ingGroupCreated( Element( real_name, lastInsertID() ) ); +} + +void TQSqlRecipeDB::createNewIngredient( const TQString &ingredientName ) +{ + TQString command; + TQString real_name = ingredientName.left( maxIngredientNameLength() ); + + command = TQString( "INSERT INTO ingredients VALUES(%2,'%1');" ).arg( escapeAndEncode( real_name ) ).arg( getNextInsertIDStr( "ingredients", "id" ) ); + TQSqlQuery ingredientToCreate( command, database ); + + emit ingredientCreated( Element( real_name, lastInsertID() ) ); +} + +void TQSqlRecipeDB::createNewRating( const TQString &rating ) +{ + TQString command; + TQString real_name = rating/*.left( maxIngredientNameLength() )*/; + + command = TQString( "INSERT INTO rating_criteria VALUES(%2,'%1');" ).arg( escapeAndEncode( real_name ) ).arg( getNextInsertIDStr( "rating_criteria", "id" ) ); + TQSqlQuery toCreate( command, database ); + + emit ratingCriteriaCreated( Element( real_name, lastInsertID() ) ); +} + +void TQSqlRecipeDB::createNewYieldType( const TQString &name ) +{ + TQString command; + TQString real_name = name.left( maxYieldTypeLength() ); + + command = TQString( "INSERT INTO yield_types VALUES(%2,'%1');" ).arg( escapeAndEncode( real_name ) ).arg( getNextInsertIDStr( "yield_types", "id" ) ); + database->exec(command); + + //emit yieldTypeCreated( Element( real_name, lastInsertID() ) ); +} + +void TQSqlRecipeDB::modIngredientGroup( int groupID, const TQString &newLabel ) +{ + TQString command; + + command = TQString( "UPDATE ingredient_groups SET name='%1' WHERE id=%2;" ).arg( escapeAndEncode( newLabel ) ).arg( groupID ); + TQSqlQuery ingredientToCreate( command, database ); + + emit ingGroupRemoved( groupID ); + emit ingGroupCreated( Element( newLabel, groupID ) ); +} + +void TQSqlRecipeDB::modIngredient( int ingredientID, const TQString &newLabel ) +{ + TQString command; + + command = TQString( "UPDATE ingredients SET name='%1' WHERE id=%2;" ).arg( escapeAndEncode( newLabel ) ).arg( ingredientID ); + TQSqlQuery ingredientToCreate( command, database ); + + emit ingredientRemoved( ingredientID ); + emit ingredientCreated( Element( newLabel, ingredientID ) ); +} + +void TQSqlRecipeDB::addUnitToIngredient( int ingredientID, int unitID ) +{ + TQString command; + + command = TQString( "INSERT INTO unit_list VALUES(%1,%2);" ).arg( ingredientID ).arg( unitID ); + TQSqlQuery ingredientToCreate( command, database ); +} + +void TQSqlRecipeDB::loadUnits( UnitList *list, Unit::Type type, int limit, int offset ) +{ + list->clear(); + + TQString command; + + command = "SELECT id,name,name_abbrev,plural,plural_abbrev,type FROM units " + +((type==Unit::All)?"":"WHERE type="+TQString::number(type)) + +" ORDER BY name" + +((limit==-1)?"":" LIMIT "+TQString::number(limit)+" OFFSET "+TQString::number(offset)); + + TQSqlQuery unitToLoad( command, database ); + + if ( unitToLoad.isActive() ) { + while ( unitToLoad.next() ) { + Unit unit; + unit.id = unitToLoad.value( 0 ).toInt(); + unit.name = unescapeAndDecode( unitToLoad.value( 1 ).toCString() ); + unit.name_abbrev = unescapeAndDecode( unitToLoad.value( 2 ).toCString() ); + unit.plural = unescapeAndDecode( unitToLoad.value( 3 ).toCString() ); + unit.plural_abbrev = unescapeAndDecode( unitToLoad.value( 4 ).toCString() ); + unit.type = (Unit::Type)unitToLoad.value( 5 ).toInt(); + list->append( unit ); + } + } +} + +void TQSqlRecipeDB::removeUnitFromIngredient( int ingredientID, int unitID ) +{ + TQString command; + + command = TQString( "DELETE FROM unit_list WHERE ingredient_id=%1 AND unit_id=%2;" ).arg( ingredientID ).arg( unitID ); + TQSqlQuery unitToRemove( command, database ); + + // Remove any recipe using that combination of ingredients also (user must have been warned before calling this function!) + + command = TQString( "SELECT r.id FROM recipes r,ingredient_list il WHERE r.id=il.recipe_id AND il.ingredient_id=%1 AND il.unit_id=%2;" ).arg( ingredientID ).arg( unitID ); + unitToRemove.exec( command ); + if ( unitToRemove.isActive() ) { + while ( unitToRemove.next() ) { + emit recipeRemoved( unitToRemove.value( 0 ).toInt() ); + database->exec( TQString( "DELETE FROM recipes WHERE id=%1;" ).arg( unitToRemove.value( 0 ).toInt() ) ); + } + } + + // Remove any ingredient in ingredient_list which has references to this unit and ingredient + command = TQString( "DELETE FROM ingredient_list WHERE ingredient_id=%1 AND unit_id=%2;" ).arg( ingredientID ).arg( unitID ); + unitToRemove.exec( command ); + + // Remove any ingredient properties from ingredient_info where the this ingredient+unit is being used (user must have been warned before calling this function!) + command = TQString( "DELETE FROM ingredient_info ii WHERE ii.ingredient_id=%1 AND ii.per_units=%2;" ).arg( ingredientID ).arg( unitID ); + unitToRemove.exec( command ); + + // Clean up ingredient_list which have no recipe that they belong to + // MySQL doesn't support subqueries until 4.1, so we'll do this the long way + // (Easy way: DELETE FROM ingredient_list WHERE recipe_id NOT IN ( SELECT id FROM recipes );) + TQStringList ids; + command = TQString( "SELECT id FROM recipes;" ); + unitToRemove.exec( command ); + if ( unitToRemove.isActive() ) { + while ( unitToRemove.next() ) { + ids << TQString::number( unitToRemove.value( 0 ).toInt() ); + } + } + command = TQString( "DELETE FROM ingredient_list WHERE recipe_id NOT IN ( %1 );" ).arg( ( ids.count() == 0 ) ? "-1" : ids.join( "," ) ); + unitToRemove.exec( command ); + + // Clean up category_list which have no recipe that they belong to + command = TQString( "DELETE FROM category_list WHERE recipe_id NOT IN ( %1 );" ).arg( ( ids.count() == 0 ) ? "-1" : ids.join( "," ) ); + unitToRemove.exec( command ); + + // Clean up ingredient_groups which have no recipe that they belong to + // MySQL doesn't support subqueries until 4.1, so we'll do this the long way + // (Easy way: DELETE FROM ingredient_groups WHERE id NOT IN ( SELECT DISTINCT(group_id) FROM ingredient_list );) + ids.clear(); + command = TQString( "SELECT DISTINCT(group_id) FROM ingredient_list;" ); + unitToRemove.exec( command ); + if ( unitToRemove.isActive() ) { + while ( unitToRemove.next() ) { + if ( unitToRemove.value( 0 ).toInt() != -1 ) + ids << TQString::number( unitToRemove.value( 0 ).toInt() ); + } + } + command = TQString( "DELETE FROM ingredient_groups WHERE id NOT IN ( %1 );" ).arg( ( ids.count() == 0 ) ? "-1" : ids.join( "," ) ); + unitToRemove.exec( command ); +} + +void TQSqlRecipeDB::removeIngredientGroup( int groupID ) +{ + TQString command; + + // First remove the ingredient + + command = TQString( "DELETE FROM ingredient_groups WHERE id=%1" ).arg( groupID ); + TQSqlQuery toDelete( command, database ); + + // Remove all the unit entries for this ingredient + + command = TQString( "UPDATE ingredient_list SET group_id='-1' WHERE group_id=%1" ).arg( groupID ); + toDelete.exec( command ); + + emit ingGroupRemoved( groupID ); +} + +void TQSqlRecipeDB::removeIngredient( int ingredientID ) +{ + TQString command; + + // First remove the ingredient + + command = TQString( "DELETE FROM ingredients WHERE id=%1;" ).arg( ingredientID ); + TQSqlQuery ingredientToDelete( command, database ); + + // Remove all the unit entries for this ingredient + + command = TQString( "DELETE FROM unit_list WHERE ingredient_id=%1;" ).arg( ingredientID ); + ingredientToDelete.exec( command ); + + // Remove any recipe using that ingredient + + command = TQString( "SELECT r.id FROM recipes r,ingredient_list il WHERE r.id=il.recipe_id AND il.ingredient_id=%1;" ).arg( ingredientID ); + ingredientToDelete.exec( command ); + if ( ingredientToDelete.isActive() ) { + while ( ingredientToDelete.next() ) { + emit recipeRemoved( ingredientToDelete.value( 0 ).toInt() ); + database->exec( TQString( "DELETE FROM recipes WHERE id=%1;" ).arg( ingredientToDelete.value( 0 ).toInt() ) ); + } + } + + // Remove any ingredient in ingredient_list which has references to this ingredient + command = TQString( "DELETE FROM ingredient_list WHERE ingredient_id=%1;" ).arg( ingredientID ); + ingredientToDelete.exec( command ); + + // Clean up ingredient_list which have no recipe that they belong to + // MySQL doesn't support subqueries until 4.1, so we'll do this the long way + // (Easy way: DELETE FROM ingredient_list WHERE recipe_id NOT IN ( SELECT id FROM recipes );) + TQStringList ids; + command = TQString( "SELECT id FROM recipes;" ); + ingredientToDelete.exec( command ); + if ( ingredientToDelete.isActive() ) { + while ( ingredientToDelete.next() ) { + ids << TQString::number( ingredientToDelete.value( 0 ).toInt() ); + } + } + command = TQString( "DELETE FROM ingredient_list WHERE recipe_id NOT IN ( %1 );" ).arg( ( ids.count() == 0 ) ? "-1" : ids.join( "," ) ); + ingredientToDelete.exec( command ); + + // Clean up category_list which have no recipe that they belong to. Same method as above + command = TQString( "DELETE FROM category_list WHERE recipe_id NOT IN ( %1 );" ).arg( ( ids.count() == 0 ) ? "-1" : ids.join( "," ) ); + ingredientToDelete.exec( command ); + + // Clean up ingredient_groups which have no recipe that they belong to + // MySQL doesn't support subqueries until 4.1, so we'll do this the long way + // (Easy way: DELETE FROM ingredient_groups WHERE id NOT IN ( SELECT DISTINCT(group_id) FROM ingredient_list );) + ids.clear(); + command = TQString( "SELECT DISTINCT(group_id) FROM ingredient_list;" ); + ingredientToDelete.exec( command ); + if ( ingredientToDelete.isActive() ) { + while ( ingredientToDelete.next() ) { + if ( ingredientToDelete.value( 0 ).toInt() != -1 ) + ids << TQString::number( ingredientToDelete.value( 0 ).toInt() ); + } + } + command = TQString( "DELETE FROM ingredient_groups WHERE id NOT IN ( %1 );" ).arg( ( ids.count() == 0 ) ? "-1" : ids.join( "," ) ); + ingredientToDelete.exec( command ); + + // Remove property list of this ingredient + command = TQString( "DELETE FROM ingredient_info WHERE ingredient_id=%1;" ).arg( ingredientID ); + ingredientToDelete.exec( command ); + + emit ingredientRemoved( ingredientID ); +} + +void TQSqlRecipeDB::removeIngredientWeight( int id ) +{ + TQString command; + + // First remove the ingredient + + command = TQString( "DELETE FROM ingredient_weights WHERE id=%1" ).arg( id ); + TQSqlQuery toDelete( command, database ); +} + +void TQSqlRecipeDB::addIngredientWeight( const Weight &w ) +{ + TQString command; + if ( w.id != -1 ) { + command = TQString( "UPDATE ingredient_weights SET ingredient_id=%1,amount=%2,unit_id=%3,weight=%4,weight_unit_id=%5,prep_method_id=%7 WHERE id=%6" ) + .arg(w.ingredientID) + .arg(w.perAmount) + .arg(w.perAmountUnitID) + .arg(w.weight) + .arg(w.weightUnitID) + .arg(w.id) + .arg(w.prepMethodID); + } + else { + command = TQString( "INSERT INTO ingredient_weights VALUES(%6,%1,%2,%3,%4,%5,%7)" ) + .arg(w.ingredientID) + .arg(w.perAmount) + .arg(w.perAmountUnitID) + .arg(w.weight) + .arg(w.weightUnitID) + .arg(getNextInsertIDStr( "ingredient_weights", "id" )) + .arg(w.prepMethodID); + } + TQSqlQuery query( command, database ); +} + +void TQSqlRecipeDB::addProperty( const TQString &name, const TQString &units ) +{ + TQString command; + TQString real_name = name.left( maxPropertyNameLength() ); + + command = TQString( "INSERT INTO ingredient_properties VALUES(%3,'%1','%2');" ) + .arg( escapeAndEncode( real_name ) ) + .arg( escapeAndEncode( units ) ) + .arg( getNextInsertIDStr( "ingredient_properties", "id" ) ); + TQSqlQuery propertyToAdd( command, database ); + + emit propertyCreated( IngredientProperty( real_name, units, lastInsertID() ) ); +} + +void TQSqlRecipeDB::loadProperties( IngredientPropertyList *list, int ingredientID ) +{ + list->clear(); + TQString command; + bool usePerUnit; + if ( ingredientID >= 0 ) // Load properties of this ingredient + { + usePerUnit = true; + command = TQString( "SELECT ip.id,ip.name,ip.units,ii.per_units,u.name,u.type,ii.amount,ii.ingredient_id FROM ingredient_properties ip, ingredient_info ii, units u WHERE ii.ingredient_id=%1 AND ii.property_id=ip.id AND ii.per_units=u.id;" ).arg( ingredientID ); + } + else if ( ingredientID == -1 ) // Load the properties of all the ingredients + { + usePerUnit = true; + command = TQString( "SELECT ip.id,ip.name,ip.units,ii.per_units,u.name,u.type,ii.amount,ii.ingredient_id FROM ingredient_properties ip, ingredient_info ii, units u WHERE ii.property_id=ip.id AND ii.per_units=u.id;" ); + } + else // Load the whole property list (just the list of possible properties, not the ingredient properties) + { + usePerUnit = false; + command = TQString( "SELECT id,name,units FROM ingredient_properties;" ); + } + + TQSqlQuery propertiesToLoad ( command, database ); + // Load the results into the list + if ( propertiesToLoad.isActive() ) { + while ( propertiesToLoad.next() ) { + IngredientProperty prop; + prop.id = propertiesToLoad.value( 0 ).toInt(); + prop.name = unescapeAndDecode( propertiesToLoad.value( 1 ).toCString() ); + prop.units = unescapeAndDecode( propertiesToLoad.value( 2 ).toCString() ); + if ( usePerUnit ) { + prop.perUnit.id = propertiesToLoad.value( 3 ).toInt(); + prop.perUnit.name = unescapeAndDecode( propertiesToLoad.value( 4 ).toCString() ); + prop.perUnit.type = (Unit::Type)propertiesToLoad.value( 5 ).toInt(); + } + + if ( ingredientID >= -1 ) + prop.amount = propertiesToLoad.value( 6 ).toDouble(); + else + prop.amount = -1; // Property is generic, not attached to an ingredient + + if ( ingredientID >= -1 ) + prop.ingredientID = propertiesToLoad.value( 7 ).toInt(); + + list->append( prop ); + } + } +} + +void TQSqlRecipeDB::changePropertyAmountToIngredient( int ingredientID, int propertyID, double amount, int per_units ) +{ + TQString command; + command = TQString( "UPDATE ingredient_info SET amount=%1 WHERE ingredient_id=%2 AND property_id=%3 AND per_units=%4;" ).arg( amount ).arg( ingredientID ).arg( propertyID ).arg( per_units ); + TQSqlQuery infoToChange( command, database ); +} + +void TQSqlRecipeDB::addPropertyToIngredient( int ingredientID, int propertyID, double amount, int perUnitsID ) +{ + TQString command; + + command = TQString( "INSERT INTO ingredient_info VALUES(%1,%2,%3,%4);" ).arg( ingredientID ).arg( propertyID ).arg( amount ).arg( perUnitsID ); + TQSqlQuery propertyToAdd( command, database ); +} + + +void TQSqlRecipeDB::removePropertyFromIngredient( int ingredientID, int propertyID, int perUnitID ) +{ + TQString command; + // remove property from ingredient info. Note that there could be duplicates with different units (per_units). Remove just the one especified. + command = TQString( "DELETE FROM ingredient_info WHERE ingredient_id=%1 AND property_id=%2 AND per_units=%3;" ).arg( ingredientID ).arg( propertyID ).arg( perUnitID ); + TQSqlQuery propertyToRemove( command, database ); +} + +void TQSqlRecipeDB::removeProperty( int propertyID ) +{ + TQString command; + + // Remove property from the ingredient_properties + command = TQString( "DELETE FROM ingredient_properties WHERE id=%1;" ).arg( propertyID ); + TQSqlQuery propertyToRemove( command, database ); + + // Remove any ingredient info that uses this property + command = TQString( "DELETE FROM ingredient_info WHERE property_id=%1;" ).arg( propertyID ); + propertyToRemove.exec( command ); + + emit propertyRemoved( propertyID ); +} + +void TQSqlRecipeDB::removeUnit( int unitID ) +{ + TQString command; + // Remove the unit first + command = TQString( "DELETE FROM units WHERE id=%1;" ).arg( unitID ); + TQSqlQuery unitToRemove( command, database ); + + //Remove the unit from ingredients using it + + command = TQString( "DELETE FROM unit_list WHERE unit_id=%1;" ).arg( unitID ); + unitToRemove.exec( command ); + + + // Remove any recipe using that unit in the ingredient list (user must have been warned before calling this function!) + + command = TQString( "SELECT r.id FROM recipes r,ingredient_list il WHERE r.id=il.recipe_id AND il.unit_id=%1;" ).arg( unitID ); + unitToRemove.exec( command ); + if ( unitToRemove.isActive() ) { + while ( unitToRemove.next() ) { + emit recipeRemoved( unitToRemove.value( 0 ).toInt() ); + database->exec( TQString( "DELETE FROM recipes WHERE id=%1;" ).arg( unitToRemove.value( 0 ).toInt() ) ); + } + } + + // Remove any ingredient in ingredient_list which has references to this unit + command = TQString( "DELETE FROM ingredient_list WHERE unit_id=%1;" ).arg( unitID ); + unitToRemove.exec( command ); + + // Clean up ingredient_list which have no recipe that they belong to + // MySQL doesn't support subqueries until 4.1, so we'll do this the long way + // (Easy way: DELETE FROM ingredient_list WHERE recipe_id NOT IN ( SELECT id FROM recipes );) + TQStringList ids; + command = TQString( "SELECT id FROM recipes;" ); + unitToRemove.exec( command ); + if ( unitToRemove.isActive() ) { + while ( unitToRemove.next() ) { + ids << TQString::number( unitToRemove.value( 0 ).toInt() ); + } + } + command = TQString( "DELETE FROM ingredient_list WHERE recipe_id NOT IN ( %1 );" ).arg( ( ids.count() == 0 ) ? "-1" : ids.join( "," ) ); + unitToRemove.exec( command ); + + // Clean up category_list which have no recipe that they belong to. Same method as above + command = TQString( "DELETE FROM category_list WHERE recipe_id NOT IN ( %1 );" ).arg( ( ids.count() == 0 ) ? "-1" : ids.join( "," ) ); + unitToRemove.exec( command ); + + // Clean up ingredient_groups which have no recipe that they belong to + // MySQL doesn't support subqueries until 4.1, so we'll do this the long way + // (Easy way: DELETE FROM ingredient_groups WHERE id NOT IN ( SELECT DISTINCT(group_id) FROM ingredient_list );) + ids.clear(); + command = TQString( "SELECT DISTINCT(group_id) FROM ingredient_list;" ); + unitToRemove.exec( command ); + if ( unitToRemove.isActive() ) { + while ( unitToRemove.next() ) { + if ( unitToRemove.value( 0 ).toInt() != -1 ) + ids << TQString::number( unitToRemove.value( 0 ).toInt() ); + } + } + command = TQString( "DELETE FROM ingredient_groups WHERE id NOT IN ( %1 );" ).arg( ( ids.count() == 0 ) ? "-1" : ids.join( "," ) ); + unitToRemove.exec( command ); + + // Remove the ingredient properties using this unit (user must be warned before calling this function) + command = TQString( "DELETE FROM ingredient_info WHERE per_units=%1;" ).arg( unitID ); + unitToRemove.exec( command ); + + // Remove the unit conversion ratios with this unit + command = TQString( "DELETE FROM units_conversion WHERE unit1_id=%1 OR unit2_id=%2;" ).arg( unitID ).arg( unitID ); + unitToRemove.exec( command ); + + // Remove associated ingredient weights + command = TQString( "DELETE FROM ingredient_weights WHERE unit_id=%1" ).arg( unitID ); + unitToRemove.exec( command ); + + emit unitRemoved( unitID ); +} + +void TQSqlRecipeDB::removePrepMethod( int prepMethodID ) +{ + TQString command; + // Remove the prep method first + command = TQString( "DELETE FROM prep_methods WHERE id=%1;" ).arg( prepMethodID ); + TQSqlQuery prepMethodToRemove( command, database ); + + // Remove any recipe using that prep method in the ingredient list (user must have been warned before calling this function!) + + command = TQString( "SELECT DISTINCT r.id FROM recipes r,ingredient_list il, prep_method_list pl WHERE r.id=il.recipe_id AND pl.ingredient_list_id=il.id AND pl.prep_method_id=%1;" ).arg( prepMethodID ); + prepMethodToRemove.exec( command ); + if ( prepMethodToRemove.isActive() ) { + while ( prepMethodToRemove.next() ) { + emit recipeRemoved( prepMethodToRemove.value( 0 ).toInt() ); + database->exec( TQString( "DELETE FROM recipes WHERE id=%1;" ).arg( prepMethodToRemove.value( 0 ).toInt() ) ); + } + } + + // Clean up ingredient_list which have no recipe that they belong to + // MySQL doesn't support subqueries until 4.1, so we'll do this the long way + // (Easy way: DELETE FROM ingredient_list WHERE recipe_id NOT IN ( SELECT id FROM recipes );) + TQStringList ids; + command = TQString( "SELECT id FROM recipes;" ); + prepMethodToRemove.exec( command ); + if ( prepMethodToRemove.isActive() ) { + while ( prepMethodToRemove.next() ) { + ids << TQString::number( prepMethodToRemove.value( 0 ).toInt() ); + } + } + + command = TQString( "DELETE FROM ingredient_list WHERE recipe_id NOT IN ( %1 );" ).arg( ( ids.count() == 0 ) ? "-1" : ids.join( "," ) ); + prepMethodToRemove.exec( command ); + + // Clean up category_list which have no recipe that they belong to. Same method as above + command = TQString( "DELETE FROM category_list WHERE recipe_id NOT IN ( %1 );" ).arg( ( ids.count() == 0 ) ? "-1" : ids.join( "," ) ); + prepMethodToRemove.exec( command ); + + // Clean up ingredient_groups which have no recipe that they belong to + // MySQL doesn't support subqueries until 4.1, so we'll do this the long way + // (Easy way: DELETE FROM ingredient_groups WHERE id NOT IN ( SELECT DISTINCT(group_id) FROM ingredient_list );) + ids.clear(); + command = TQString( "SELECT DISTINCT(group_id) FROM ingredient_list;" ); + prepMethodToRemove.exec( command ); + if ( prepMethodToRemove.isActive() ) { + while ( prepMethodToRemove.next() ) { + if ( prepMethodToRemove.value( 0 ).toInt() != -1 ) + ids << TQString::number( prepMethodToRemove.value( 0 ).toInt() ); + } + } + command = TQString( "DELETE FROM ingredient_groups WHERE id NOT IN ( %1 );" ).arg( ( ids.count() == 0 ) ? "-1" : ids.join( "," ) ); + prepMethodToRemove.exec( command ); + + emit prepMethodRemoved( prepMethodID ); +} + + +void TQSqlRecipeDB::createNewUnit( const Unit &unit ) +{ + TQString real_name = unit.name.left( maxUnitNameLength() ).stripWhiteSpace(); + TQString real_plural = unit.plural.left( maxUnitNameLength() ).stripWhiteSpace(); + TQString real_name_abbrev = unit.name_abbrev.left( maxUnitNameLength() ).stripWhiteSpace(); + TQString real_plural_abbrev = unit.plural_abbrev.left( maxUnitNameLength() ).stripWhiteSpace(); + + Unit new_unit( real_name, real_plural ); + new_unit.name_abbrev = real_name_abbrev; + new_unit.plural_abbrev = real_plural_abbrev; + new_unit.type = unit.type; + + if ( real_name.isEmpty() ) + real_name = real_plural; + else if ( real_plural.isEmpty() ) + real_plural = real_name; + + if ( real_name_abbrev.isEmpty() ) + real_name_abbrev = "NULL"; + else + real_name_abbrev = "'"+escapeAndEncode(real_name_abbrev)+"'"; + if ( real_plural_abbrev.isEmpty() ) + real_plural_abbrev = "NULL"; + else + real_plural_abbrev = "'"+escapeAndEncode(real_plural_abbrev)+"'"; + + + TQString command = "INSERT INTO units VALUES(" + getNextInsertIDStr( "units", "id" ) + + ",'" + escapeAndEncode( real_name ) + + "'," + real_name_abbrev + + ",'" + escapeAndEncode( real_plural ) + + "'," + real_plural_abbrev + + "," + TQString::number(unit.type) + + ");"; + + TQSqlQuery unitToCreate( command, database ); + + new_unit.id = lastInsertID(); + emit unitCreated( new_unit ); +} + + +void TQSqlRecipeDB::modUnit( const Unit &unit ) +{ + TQSqlQuery unitQuery( TQString::null, database ); + + TQString real_name = unit.name.left( maxUnitNameLength() ).stripWhiteSpace(); + TQString real_plural = unit.plural.left( maxUnitNameLength() ).stripWhiteSpace(); + TQString real_name_abbrev = unit.name_abbrev.left( maxUnitNameLength() ).stripWhiteSpace(); + TQString real_plural_abbrev = unit.plural_abbrev.left( maxUnitNameLength() ).stripWhiteSpace(); + + Unit newUnit( real_name, real_plural, unit.id ); + newUnit.type = unit.type; + newUnit.name_abbrev = real_name_abbrev; + newUnit.plural_abbrev = real_plural_abbrev; + + if ( real_name_abbrev.isEmpty() ) + real_name_abbrev = "NULL"; + else + real_name_abbrev = "'"+escapeAndEncode(real_name_abbrev)+"'"; + if ( real_plural_abbrev.isEmpty() ) + real_plural_abbrev = "NULL"; + else + real_plural_abbrev = "'"+escapeAndEncode(real_plural_abbrev)+"'"; + + TQString command = TQString("UPDATE units SET name='%1',name_abbrev=%2,plural='%3',plural_abbrev=%4,type=%6 WHERE id='%5'") + .arg(escapeAndEncode(real_name)) + .arg(real_name_abbrev) + .arg(escapeAndEncode(real_plural)) + .arg(real_plural_abbrev) + .arg(unit.id) + .arg(unit.type); + unitQuery.exec( command ); + + emit unitRemoved( unit.id ); + emit unitCreated( newUnit ); +} + +void TQSqlRecipeDB::findUseOfIngGroupInRecipes( ElementList *results, int groupID ) +{ + TQString command = TQString( "SELECT DISTINCT r.id,r.title FROM recipes r,ingredient_list il WHERE r.id=il.recipe_id AND il.group_id=%1" ).arg( groupID ); + TQSqlQuery query( command, database ); + + // Populate data + if ( query.isActive() ) { + while ( query.next() ) { + Element recipe; + recipe.id = query.value( 0 ).toInt(); + recipe.name = unescapeAndDecode( query.value( 1 ).toCString() ); + results->append( recipe ); + } + } +} + +void TQSqlRecipeDB::findUseOfCategoryInRecipes( ElementList *results, int catID ) +{ + TQString command = TQString( "SELECT r.id,r.title FROM recipes r,category_list cl WHERE r.id=cl.recipe_id AND cl.category_id=%1" ).arg( catID ); + TQSqlQuery query( command, database ); + + // Populate data + if ( query.isActive() ) { + while ( query.next() ) { + Element recipe; + recipe.id = query.value( 0 ).toInt(); + recipe.name = unescapeAndDecode( query.value( 1 ).toCString() ); + results->append( recipe ); + } + } + + //recursively find dependenacies in subcategories + command = TQString( "SELECT id FROM categories WHERE parent_id=%1" ).arg( catID ); + TQSqlQuery findDeps = database->exec( command ); + if ( findDeps.isActive() ) { + while ( findDeps.next() ) { + findUseOfCategoryInRecipes(results,findDeps.value( 0 ).toInt() ); + } + } +} + +void TQSqlRecipeDB::findUseOfAuthorInRecipes( ElementList *results, int authorID ) +{ + TQString command = TQString( "SELECT r.id,r.title FROM recipes r,author_list al WHERE r.id=al.recipe_id AND al.author_id=%1" ).arg( authorID ); + TQSqlQuery query( command, database ); + + // Populate data + if ( query.isActive() ) { + while ( query.next() ) { + Element recipe; + recipe.id = query.value( 0 ).toInt(); + recipe.name = unescapeAndDecode( query.value( 1 ).toCString() ); + results->append( recipe ); + } + } +} + +void TQSqlRecipeDB::loadUnitRatios( UnitRatioList *ratioList, Unit::Type type ) +{ + ratioList->clear(); + + TQString command; + if ( type == Unit::All ) + command = "SELECT unit1_id,unit2_id,ratio FROM units_conversion"; + else + command = "SELECT unit1_id,unit2_id,ratio FROM units_conversion,units unit1,units unit2 WHERE unit1_id=unit1.id AND unit1.type="+TQString::number(type)+" AND unit2_id=unit2.id AND unit2.type="+TQString::number(type); + TQSqlQuery ratiosToLoad( command, database ); + + if ( ratiosToLoad.isActive() ) { + while ( ratiosToLoad.next() ) { + UnitRatio ratio; + ratio.uID1 = ratiosToLoad.value( 0 ).toInt(); + ratio.uID2 = ratiosToLoad.value( 1 ).toInt(); + ratio.ratio = ratiosToLoad.value( 2 ).toDouble(); + ratioList->add( ratio ); + } + } +} + +void TQSqlRecipeDB::saveUnitRatio( const UnitRatio *ratio ) +{ + TQString command; + + // Check if it's a new ratio or it exists already. + command = TQString( "SELECT * FROM units_conversion WHERE unit1_id=%1 AND unit2_id=%2" ).arg( ratio->uID1 ).arg( ratio->uID2 ); // Find ratio between units + + TQSqlQuery ratioFound( command, database ); // Find the entries + bool newRatio = ( ratioFound.size() == 0 ); + + if ( newRatio ) + command = TQString( "INSERT INTO units_conversion VALUES(%1,%2,%3);" ).arg( ratio->uID1 ).arg( ratio->uID2 ).arg( ratio->ratio ); + else + command = TQString( "UPDATE units_conversion SET ratio=%3 WHERE unit1_id=%1 AND unit2_id=%2" ).arg( ratio->uID1 ).arg( ratio->uID2 ).arg( ratio->ratio ); + + ratioFound.exec( command ); // Enter the new ratio +} + +void TQSqlRecipeDB::removeUnitRatio( int unitID1, int unitID2 ) +{ + database->exec(TQString( "DELETE FROM units_conversion WHERE unit1_id=%1 AND unit2_id=%2" ).arg( unitID1 ).arg( unitID2 )); +} + +double TQSqlRecipeDB::unitRatio( int unitID1, int unitID2 ) +{ + + if ( unitID1 == unitID2 ) + return ( 1.0 ); + TQString command; + + command = TQString( "SELECT ratio FROM units_conversion WHERE unit1_id=%1 AND unit2_id=%2;" ).arg( unitID1 ).arg( unitID2 ); + TQSqlQuery ratioToLoad( command, database ); + + if ( ratioToLoad.isActive() && ratioToLoad.next() ) + return ( ratioToLoad.value( 0 ).toDouble() ); + else + return ( -1 ); +} + +double TQSqlRecipeDB::ingredientWeight( const Ingredient &ing, bool *wasApproximated ) +{ + TQString command = TQString( "SELECT amount,weight,prep_method_id,unit_id FROM ingredient_weights WHERE ingredient_id=%1 AND (unit_id=%2 OR weight_unit_id=%3)" ) + .arg( ing.ingredientID ) + .arg( ing.units.id ).arg( ing.units.id ); + + TQSqlQuery query( command, database ); + + if ( query.isActive() ) { + //store the amount for the entry with no prep method. If no other suitable entry is found, we'll guesstimate + //the weight using this entry + double convertedAmount = -1; + while ( query.next() ) { + int prepMethodID = query.value( 2 ).toInt(); + + if ( ing.prepMethodList.containsId( prepMethodID ) ) { + if ( wasApproximated ) *wasApproximated = false; + double amount = query.value( 0 ).toDouble(); + + //'per_amount' -> 'weight' conversion + if ( query.value( 3 ).toInt() == ing.units.id ) + convertedAmount = query.value( 1 ).toDouble() * ing.amount / amount; + //'weight' -> 'per_amount' conversion + else + convertedAmount = amount * ing.amount / query.value( 1 ).toDouble(); + + return convertedAmount; + } + if ( prepMethodID == -1 ) { + //'per_amount' -> 'weight' conversion + if ( query.value( 3 ).toInt() == ing.units.id ) + convertedAmount = query.value( 1 ).toDouble() * ing.amount / query.value( 0 ).toDouble(); + //'weight' -> 'per_amount' conversion + else + convertedAmount = query.value( 0 ).toDouble() * ing.amount / query.value( 1 ).toDouble(); + } + } + //no matching prep method found, use entry without a prep method if there was one + if ( convertedAmount > 0 ) { + if ( wasApproximated ) *wasApproximated = true; + kdDebug()<<"Prep method given, but no weight entry found that uses that prep method. I'm fudging the weight with an entry without a prep method."<<endl; + + return convertedAmount; + } + } + return -1; +} + +WeightList TQSqlRecipeDB::ingredientWeightUnits( int ingID ) +{ + WeightList list; + + TQString command = TQString( "SELECT id,amount,unit_id,weight,weight_unit_id,prep_method_id FROM ingredient_weights WHERE ingredient_id=%1" ).arg( ingID ); + TQSqlQuery query( command, database ); + if ( query.isActive() ) { + while ( query.next() ) { + Weight w; + w.id = query.value(0).toInt(); + w.perAmount = query.value(1).toDouble(); + w.perAmountUnitID = query.value(2).toInt(); + w.weight = query.value(3).toDouble(); + w.weightUnitID = query.value(4).toInt(); + w.prepMethodID = query.value(5).toInt(); + w.ingredientID = ingID; + list.append(w); + } + } + + return list; +} + +//Finds data dependant on this Ingredient/Unit combination +void TQSqlRecipeDB::findIngredientUnitDependancies( int ingredientID, int unitID, ElementList *recipes, ElementList *ingredientInfo ) +{ + + // Recipes using that combination + + TQString command = TQString( "SELECT DISTINCT r.id,r.title FROM recipes r,ingredient_list il WHERE r.id=il.recipe_id AND il.ingredient_id=%1 AND il.unit_id=%2;" ).arg( ingredientID ).arg( unitID ); + TQSqlQuery unitToRemove( command, database ); + loadElementList( recipes, &unitToRemove ); + // Ingredient info using that combination + command = TQString( "SELECT i.name,ip.name,ip.units,u.name FROM ingredients i, ingredient_info ii, ingredient_properties ip, units u WHERE i.id=ii.ingredient_id AND ii.ingredient_id=%1 AND ii.per_units=%2 AND ii.property_id=ip.id AND ii.per_units=u.id;" ).arg( ingredientID ).arg( unitID ); + + unitToRemove.exec( command ); + loadPropertyElementList( ingredientInfo, &unitToRemove ); +} + +void TQSqlRecipeDB::findIngredientDependancies( int ingredientID, ElementList *recipes ) +{ + TQString command = TQString( "SELECT DISTINCT r.id,r.title FROM recipes r,ingredient_list il WHERE r.id=il.recipe_id AND il.ingredient_id=%1" ).arg( ingredientID ); + + TQSqlQuery ingredientToRemove( command, database ); + loadElementList( recipes, &ingredientToRemove ); +} + + + +//Finds data dependant on the removal of this Unit +void TQSqlRecipeDB::findUnitDependancies( int unitID, ElementList *properties, ElementList *recipes, ElementList *weights ) +{ + + // Ingredient-Info (ingredient->property) using this Unit + + TQString command = TQString( "SELECT i.name,ip.name,ip.units,u.name FROM ingredients i, ingredient_info ii, ingredient_properties ip, units u WHERE i.id=ii.ingredient_id AND ii.per_units=%1 AND ii.property_id=ip.id AND ii.per_units=u.id;" ).arg( unitID ); + TQSqlQuery unitToRemove( command, database ); + loadPropertyElementList( properties, &unitToRemove ); + + // Recipes using this Unit + command = TQString( "SELECT DISTINCT r.id,r.title FROM recipes r,ingredient_list il WHERE r.id=il.recipe_id AND il.unit_id=%1;" ).arg( unitID ); // Without "DISTINCT" we get duplicates since ingredient_list has no unique recipe_id's + unitToRemove.exec( command ); + loadElementList( recipes, &unitToRemove ); + + // Weights using this unit + command = TQString( "SELECT i.name,weight_u.name,per_u.name,w.prep_method_id FROM ingredients i,ingredient_weights w,units weight_u,units per_u WHERE i.id=w.ingredient_id AND w.unit_id=per_u.id AND w.weight_unit_id=weight_u.id AND (weight_u.id=%1 OR per_u.id=%2)" ) + .arg( unitID ) + .arg( unitID ); + unitToRemove.exec( command ); + if ( unitToRemove.isActive() ) { + while ( unitToRemove.next() ) { + Element el; + + TQString ingName = unescapeAndDecode( unitToRemove.value( 0 ).toCString() ); + TQString weightUnit = unescapeAndDecode( unitToRemove.value( 1 ).toCString() ); + TQString perUnit = unescapeAndDecode( unitToRemove.value( 2 ).toCString() ); + + int prepID = unitToRemove.value( 3 ).toInt(); + TQString prep; + if ( prepID != -1 ) { + command = TQString( "SELECT p.name FROM prep_methods p, ingredient_weights w WHERE p.id = w.prep_method_id AND w.prep_method_id=%1" ) + .arg( prepID ); + TQSqlQuery query( command, database ); + if ( query.isActive() && query.first() ) + prep = unescapeAndDecode( query.value( 0 ).toCString() ); + } + + el.name = TQString( i18n("In ingredient '%1': weight [%2/%3%4]") ).arg( ingName ).arg( weightUnit ).arg( perUnit ).arg( (prepID == -1)?TQString::null:"; "+prep ); + weights->append( el ); + } + } + +} + +void TQSqlRecipeDB::findPrepMethodDependancies( int prepMethodID, ElementList *recipes ) +{ + //get just the ids first so that we can use DISTINCT + TQString command = TQString( "SELECT DISTINCT r.id FROM recipes r,ingredient_list il, prep_method_list pl WHERE r.id=il.recipe_id AND pl.ingredient_list_id=il.id AND pl.prep_method_id=%1;" ).arg( prepMethodID ); + + TQStringList ids; + TQSqlQuery query( command, database ); + if ( query.isActive() ) { + while ( query.next() ) { + ids << TQString::number(query.value( 0 ).toInt()); + } + } + + //now get the titles of the ids + command = TQString( "SELECT r.id, r.title FROM recipes r WHERE r.id IN ("+ids.join(",")+")" ); + TQSqlQuery prepMethodToRemove( command, database ); + loadElementList( recipes, &prepMethodToRemove ); +} + + +void TQSqlRecipeDB::loadElementList( ElementList *elList, TQSqlQuery *query ) +{ + if ( query->isActive() ) { + while ( query->next() ) { + Element el; + el.id = query->value( 0 ).toInt(); + el.name = unescapeAndDecode( query->value( 1 ).toCString() ); + elList->append( el ); + } + } +} +// See function "findUnitDependancies" for use +void TQSqlRecipeDB::loadPropertyElementList( ElementList *elList, TQSqlQuery *query ) +{ + if ( query->isActive() ) { + while ( query->next() ) { + Element el; + el.id = -1; // There's no ID for the ingredient-property combination + TQString ingName = unescapeAndDecode( query->value( 0 ).toCString() ); + TQString propName = unescapeAndDecode( query->value( 1 ).toCString() ); + TQString propUnits = unescapeAndDecode( query->value( 2 ).toCString() ); + TQString propPerUnits = unescapeAndDecode( query->value( 3 ).toCString() ); + + el.name = TQString( i18n("In ingredient '%1': property \"%2\" [%3/%4]") ).arg( ingName ).arg( propName ).arg( propUnits ).arg( propPerUnits ); + elList->append( el ); + } + } +} + + +//The string going into the database is utf8 text interpreted as latin1 +TQString TQSqlRecipeDB::escapeAndEncode( const TQString &s ) const +{ + TQString s_escaped = s; + + s_escaped.replace ( "'", "\\'" ); + s_escaped.replace ( ";", "\";@" ); // Small trick for only for parsing later on + + return TQString::fromLatin1( s_escaped.utf8() ); +} + +//The string coming out of the database is utf8 text, interpreted as though latin1. Calling fromUtf8() on this gives us back the original utf8. +TQString TQSqlRecipeDB::unescapeAndDecode( const TQCString &s ) const +{ + return TQString::fromUtf8( s ).replace( "\";@", ";" ); // Use unicode encoding +} + +bool TQSqlRecipeDB::ingredientContainsUnit( int ingredientID, int unitID ) +{ + TQString command = TQString( "SELECT * FROM unit_list WHERE ingredient_id= %1 AND unit_id=%2;" ).arg( ingredientID ).arg( unitID ); + TQSqlQuery recipeToLoad( command, database ); + if ( recipeToLoad.isActive() ) { + return ( recipeToLoad.size() > 0 ); + } + return false; +} + +bool TQSqlRecipeDB::ingredientContainsProperty( int ingredientID, int propertyID, int perUnitsID ) +{ + TQString command = TQString( "SELECT * FROM ingredient_info WHERE ingredient_id=%1 AND property_id=%2 AND per_units=%3;" ).arg( ingredientID ).arg( propertyID ).arg( perUnitsID ); + TQSqlQuery recipeToLoad( command, database ); + if ( recipeToLoad.isActive() ) { + return ( recipeToLoad.size() > 0 ); + } + return false; +} + +TQString TQSqlRecipeDB::categoryName( int ID ) +{ + TQString command = TQString( "SELECT name FROM categories WHERE id=%1;" ).arg( ID ); + TQSqlQuery toLoad( command, database ); + if ( toLoad.isActive() && toLoad.next() ) // Go to the first record (there should be only one anyway. + return ( unescapeAndDecode( toLoad.value( 0 ).toCString() ) ); + + return ( TQString::null ); +} + +TQString TQSqlRecipeDB::ingredientName( int ID ) +{ + TQString command = TQString( "SELECT name FROM ingredients WHERE id=%1" ).arg( ID ); + TQSqlQuery toLoad( command, database ); + if ( toLoad.isActive() && toLoad.next() ) // Go to the first record (there should be only one anyway. + return ( unescapeAndDecode( toLoad.value( 0 ).toCString() ) ); + + return ( TQString::null ); +} + +TQString TQSqlRecipeDB::prepMethodName( int ID ) +{ + TQString command = TQString( "SELECT name FROM prep_methods WHERE id=%1" ).arg( ID ); + TQSqlQuery toLoad( command, database ); + if ( toLoad.isActive() && toLoad.next() ) // Go to the first record (there should be only one anyway. + return ( unescapeAndDecode( toLoad.value( 0 ).toCString() ) ); + + return ( TQString::null ); +} + +IngredientProperty TQSqlRecipeDB::propertyName( int ID ) +{ + TQString command = TQString( "SELECT name,units FROM ingredient_properties WHERE id=%1;" ).arg( ID ); + TQSqlQuery toLoad( command, database ); + if ( toLoad.isActive() && toLoad.next() ) { // Go to the first record (there should be only one anyway. + return ( IngredientProperty( unescapeAndDecode( toLoad.value( 0 ).toCString() ), unescapeAndDecode( toLoad.value( 1 ).toCString() ), ID ) ); + } + + return ( IngredientProperty( TQString::null, TQString::null ) ); +} + +Unit TQSqlRecipeDB::unitName( int ID ) +{ + TQString command = TQString( "SELECT name,plural,name_abbrev,plural_abbrev,type FROM units WHERE id=%1" ).arg( ID ); + TQSqlQuery toLoad( command, database ); + if ( toLoad.isActive() && toLoad.next() ) { // Go to the first record (there should be only one anyway. + Unit unit( unescapeAndDecode( toLoad.value( 0 ).toCString() ), unescapeAndDecode( toLoad.value( 1 ).toCString() ) ); + + //if we don't have both name and plural, use what we have as both + if ( unit.name.isEmpty() ) + unit.name = unit.plural; + else if ( unit.plural.isEmpty() ) + unit.plural = unit.name; + + unit.name_abbrev = unescapeAndDecode( toLoad.value( 2 ).toCString() ); + unit.plural_abbrev = unescapeAndDecode( toLoad.value( 3 ).toCString() ); + unit.type = (Unit::Type) toLoad.value( 4 ).toInt(); + unit.id = ID; + + return unit; + } + + return Unit(); +} + +int TQSqlRecipeDB::getCount( const TQString &table_name ) +{ + m_command = "SELECT COUNT(1) FROM "+table_name; + TQSqlQuery count( m_command, database ); + if ( count.isActive() && count.next() ) { // Go to the first record (there should be only one anyway. + return count.value( 0 ).toInt(); + } + + return -1; +} + +int TQSqlRecipeDB::categoryTopLevelCount() +{ + m_command = "SELECT COUNT(1) FROM categories WHERE parent_id='-1'"; + TQSqlQuery count( m_command, database ); + if ( count.isActive() && count.next() ) { // Go to the first record (there should be only one anyway. + return count.value( 0 ).toInt(); + } + + return -1; +} + +bool TQSqlRecipeDB::checkIntegrity( void ) +{ + + + // Check existence of the necessary tables (the database may be created, but empty) + TQStringList tables; + tables << "ingredient_info" << "ingredient_list" << "ingredient_properties" << "ingredient_weights" << "ingredients" << "recipes" << "unit_list" << "units" << "units_conversion" << "categories" << "category_list" << "authors" << "author_list" << "db_info" << "prep_methods" << "ingredient_groups" << "yield_types" << "prep_method_list" << "ratings" << "rating_criteria" << "rating_criterion_list"; + + TQStringList existingTableList = database->tables(); + for ( TQStringList::Iterator it = tables.begin(); it != tables.end(); ++it ) { + bool found = false; + + for ( TQStringList::Iterator ex_it = existingTableList.begin(); ( ( ex_it != existingTableList.end() ) && ( !found ) ); ++ex_it ) { + found = ( *ex_it == *it ); + } + + if ( !found ) { + kdDebug() << "Recreating missing table: " << *it << "\n"; + createTable( *it ); + } + } + + TQStringList newTableList = database->tables(); + if ( newTableList.isEmpty() ) + return false; + + + // Check for older versions, and port + + kdDebug() << "Checking database version...\n"; + float version = databaseVersion(); + kdDebug() << "version found... " << version << " \n"; + kdDebug() << "latest version... " << latestDBVersion() << endl; + if ( int( tqRound( databaseVersion() * 1e5 ) ) < int( tqRound( latestDBVersion() * 1e5 ) ) ) { //correct for float's imprecision + switch ( KMessageBox::questionYesNo( 0, i18n( "<!doc>The database was created with a previous version of Krecipes. Would you like Krecipes to update this database to work with this version of Krecipes? Depending on the number of recipes and amount of data, this could take some time.<br><br><b>Warning: After updating, this database will no longer be compatible with previous versions of Krecipes.<br><br>Cancelling this operation may result in corrupting the database.</b>" ) ) ) { + case KMessageBox::Yes: + emit progressBegin(0,TQString::null,i18n("Porting database structure..."),50); + portOldDatabases( version ); + emit progressDone(); + break; + case KMessageBox::No: + return false; + } + } + + return true; +} + +void TQSqlRecipeDB::splitCommands( TQString& s, TQStringList& sl ) +{ + sl = TQStringList::split( TQRegExp( ";{1}(?!@)" ), s ); +} + +void TQSqlRecipeDB::portOldDatabases( float /* version */ ) +{} + +float TQSqlRecipeDB::databaseVersion( void ) +{ + + TQString command = "SELECT ver FROM db_info"; + TQSqlQuery dbVersion( command, database ); + + if ( dbVersion.isActive() && dbVersion.next() ) + return ( dbVersion.value( 0 ).toDouble() ); // There should be only one (or none for old DB) element, so go to first + else + return ( 0.2 ); // if table is empty, assume oldest (0.2), and port +} + +void TQSqlRecipeDB::loadRatingCriterion( ElementList *list, int limit, int offset ) +{ + list->clear(); + + TQString command = "SELECT id,name FROM rating_criteria ORDER BY name" + +((limit==-1)?"":" LIMIT "+TQString::number(limit)+" OFFSET "+TQString::number(offset)); + TQSqlQuery toLoad( command, database ); + if ( toLoad.isActive() ) { + while ( toLoad.next() ) { + Element el; + el.id = toLoad.value( 0 ).toInt(); + el.name = unescapeAndDecode( toLoad.value( 1 ).toCString() ); + list->append( el ); + } + } +} + +void TQSqlRecipeDB::loadCategories( ElementList *list, int limit, int offset ) +{ + list->clear(); + + m_command = "SELECT id,name FROM categories ORDER BY name" + +((limit==-1)?"":" LIMIT "+TQString::number(limit)+" OFFSET "+TQString::number(offset)); + TQSqlQuery categoryToLoad( m_command, database ); + if ( categoryToLoad.isActive() ) { + while ( categoryToLoad.next() ) { + Element el; + el.id = categoryToLoad.value( 0 ).toInt(); + el.name = unescapeAndDecode( categoryToLoad.value( 1 ).toCString() ); + list->append( el ); + } + } +} + +void TQSqlRecipeDB::loadCategories( CategoryTree *list, int limit, int offset, int parent_id, bool recurse ) +{ + TQString limit_str; + if ( parent_id == -1 ) { + emit progressBegin(0,TQString::null,i18n("Loading category list")); + list->clear(); + + //only limit the number of top-level categories + limit_str = (limit==-1)?"":" LIMIT "+TQString::number(limit)+" OFFSET "+TQString::number(offset); + } + + m_command = "SELECT id,name,parent_id FROM categories WHERE parent_id='"+TQString::number(parent_id)+"' ORDER BY name "+limit_str; + + TQSqlQuery categoryToLoad( TQString::null, database ); + //categoryToLoad.setForwardOnly(true); //FIXME? Subcategories aren't loaded if this is enabled, even though we only go forward + + categoryToLoad.exec(m_command); + + if ( categoryToLoad.isActive() ) { + while ( categoryToLoad.next() ) { + emit progress(); + + int id = categoryToLoad.value( 0 ).toInt(); + Element el; + el.id = id; + el.name = unescapeAndDecode( categoryToLoad.value( 1 ).toCString() ); + CategoryTree *list_child = list->add( el ); + + if ( recurse ) { + //TQTime dbg_timer; dbg_timer.start(); kdDebug()<<" calling TQSqlRecipeDB::loadCategories"<<endl; + loadCategories( list_child, -1, -1, id ); //limit and offset won't be used + // kdDebug()<<" done in "<<dbg_timer.elapsed()<<" ms"<<endl; + } + } + } + + if ( parent_id == -1 ) + emit progressDone(); +} + +void TQSqlRecipeDB::createNewCategory( const TQString &categoryName, int parent_id ) +{ + TQString command; + TQString real_name = categoryName.left( maxCategoryNameLength() ); + + command = TQString( "INSERT INTO categories VALUES(%3,'%1',%2);" ) + .arg( escapeAndEncode( real_name ) ) + .arg( parent_id ) + .arg( getNextInsertIDStr( "categories", "id" ) ); + TQSqlQuery categoryToCreate( command, database ); + + emit categoryCreated( Element( real_name, lastInsertID() ), parent_id ); +} + +void TQSqlRecipeDB::modCategory( int categoryID, const TQString &newLabel ) +{ + TQString command = TQString( "UPDATE categories SET name='%1' WHERE id=%2;" ).arg( escapeAndEncode( newLabel ) ).arg( categoryID ); + TQSqlQuery categoryToUpdate( command, database ); + + emit categoryModified( Element( newLabel, categoryID ) ); +} + +void TQSqlRecipeDB::modCategory( int categoryID, int new_parent_id ) +{ + TQString command = TQString( "UPDATE categories SET parent_id=%1 WHERE id=%2;" ).arg( new_parent_id ).arg( categoryID ); + TQSqlQuery categoryToUpdate( command, database ); + + emit categoryModified( categoryID, new_parent_id ); +} + +void TQSqlRecipeDB::removeCategory( int categoryID ) +{ + TQString command; + + command = TQString( "DELETE FROM categories WHERE id=%1;" ).arg( categoryID ); + TQSqlQuery categoryToRemove( command, database ); + + command = TQString( "DELETE FROM category_list WHERE category_id=%1;" ).arg( categoryID ); + categoryToRemove.exec( command ); + + //recursively delete subcategories + command = TQString( "SELECT id FROM categories WHERE parent_id=%1;" ).arg( categoryID ); + categoryToRemove.exec( command ); + if ( categoryToRemove.isActive() ) { + while ( categoryToRemove.next() ) { + removeCategory( categoryToRemove.value( 0 ).toInt() ); + } + } + + emit categoryRemoved( categoryID ); +} + + +void TQSqlRecipeDB::loadAuthors( ElementList *list, int limit, int offset ) +{ + list->clear(); + TQString command = "SELECT id,name FROM authors ORDER BY name" + +((limit==-1)?"":" LIMIT "+TQString::number(limit)+" OFFSET "+TQString::number(offset)); + TQSqlQuery authorToLoad( command, database ); + if ( authorToLoad.isActive() ) { + while ( authorToLoad.next() ) { + Element el; + el.id = authorToLoad.value( 0 ).toInt(); + el.name = unescapeAndDecode( authorToLoad.value( 1 ).toCString() ); + list->append( el ); + } + } +} + +void TQSqlRecipeDB::createNewAuthor( const TQString &authorName ) +{ + TQString command; + TQString real_name = authorName.left( maxAuthorNameLength() ); + + command = TQString( "INSERT INTO authors VALUES(%2,'%1');" ).arg( escapeAndEncode( real_name ) ).arg( getNextInsertIDStr( "authors", "id" ) ); + TQSqlQuery authorToCreate( command, database ); + + emit authorCreated( Element( real_name, lastInsertID() ) ); +} + +void TQSqlRecipeDB::modAuthor( int authorID, const TQString &newLabel ) +{ + TQString command; + + command = TQString( "UPDATE authors SET name='%1' WHERE id=%2;" ).arg( escapeAndEncode( newLabel ) ).arg( authorID ); + TQSqlQuery authorToCreate( command, database ); + + emit authorRemoved( authorID ); + emit authorCreated( Element( newLabel, authorID ) ); +} + +void TQSqlRecipeDB::removeAuthor( int authorID ) +{ + TQString command; + + command = TQString( "DELETE FROM authors WHERE id=%1;" ).arg( authorID ); + TQSqlQuery authorToRemove( command, database ); + + emit authorRemoved( authorID ); +} + +int TQSqlRecipeDB::findExistingAuthorByName( const TQString& name ) +{ + TQString search_str = escapeAndEncode( name.left( maxAuthorNameLength() ) ); //truncate to the maximum size db holds + + TQString command = TQString( "SELECT id FROM authors WHERE name LIKE '%1';" ).arg( search_str ); + TQSqlQuery elementToLoad( command, database ); // Run the query + int id = -1; + + if ( elementToLoad.isActive() && elementToLoad.first() ) + id = elementToLoad.value( 0 ).toInt(); + + return id; +} + +int TQSqlRecipeDB::findExistingCategoryByName( const TQString& name ) +{ + TQString search_str = escapeAndEncode( name.left( maxCategoryNameLength() ) ); //truncate to the maximum size db holds + + TQString command = TQString( "SELECT id FROM categories WHERE name LIKE '%1';" ).arg( search_str ); + TQSqlQuery elementToLoad( command, database ); // Run the query + int id = -1; + + if ( elementToLoad.isActive() && elementToLoad.first() ) + id = elementToLoad.value( 0 ).toInt(); + + return id; +} + +int TQSqlRecipeDB::findExistingIngredientGroupByName( const TQString& name ) +{ + TQString search_str = escapeAndEncode( name.left( maxIngGroupNameLength() ) ); //truncate to the maximum size db holds + + TQString command = TQString( "SELECT id FROM ingredient_groups WHERE name LIKE '%1';" ).arg( search_str ); + TQSqlQuery elementToLoad( command, database ); // Run the query + int id = -1; + + if ( elementToLoad.isActive() && elementToLoad.first() ) + id = elementToLoad.value( 0 ).toInt(); + + return id; +} + +int TQSqlRecipeDB::findExistingIngredientByName( const TQString& name ) +{ + TQString search_str = escapeAndEncode( name.left( maxIngredientNameLength() ) ); //truncate to the maximum size db holds + + TQString command = TQString( "SELECT id FROM ingredients WHERE name LIKE '%1';" ).arg( search_str ); + TQSqlQuery elementToLoad( command, database ); // Run the query + int id = -1; + + if ( elementToLoad.isActive() && elementToLoad.first() ) + id = elementToLoad.value( 0 ).toInt(); + + return id; +} + +int TQSqlRecipeDB::findExistingPrepByName( const TQString& name ) +{ + TQString search_str = escapeAndEncode( name.left( maxPrepMethodNameLength() ) ); //truncate to the maximum size db holds + + TQString command = TQString( "SELECT id FROM prep_methods WHERE name LIKE '%1';" ).arg( search_str ); + TQSqlQuery elementToLoad( command, database ); // Run the query + int id = -1; + + if ( elementToLoad.isActive() && elementToLoad.first() ) + id = elementToLoad.value( 0 ).toInt(); + + return id; +} + +int TQSqlRecipeDB::findExistingPropertyByName( const TQString& name ) +{ + TQString search_str = escapeAndEncode( name.left( maxPropertyNameLength() ) ); //truncate to the maximum size db holds + + TQString command = TQString( "SELECT id FROM ingredient_properties WHERE name LIKE '%1';" ).arg( search_str ); + TQSqlQuery elementToLoad( command, database ); // Run the query + int id = -1; + + if ( elementToLoad.isActive() && elementToLoad.first() ) + id = elementToLoad.value( 0 ).toInt(); + + return id; +} + +int TQSqlRecipeDB::findExistingUnitByName( const TQString& name ) +{ + TQString search_str = escapeAndEncode( name.left( maxUnitNameLength() ) ); //truncate to the maximum size db holds + + TQString command = "SELECT id FROM units WHERE name LIKE '" + search_str + + "' OR plural LIKE '" + search_str + + "' OR name_abbrev LIKE '" + search_str + + "' OR plural_abbrev LIKE '" + search_str + + "'"; + + TQSqlQuery elementToLoad( command, database ); // Run the query + int id = -1; + + if ( elementToLoad.isActive() && elementToLoad.first() ) + id = elementToLoad.value( 0 ).toInt(); + + return id; +} + +int TQSqlRecipeDB::findExistingRatingByName( const TQString& name ) +{ + TQString search_str = escapeAndEncode( name ); //truncate to the maximum size db holds + + TQString command = TQString( "SELECT id FROM rating_criteria WHERE name LIKE '%1'" ).arg( search_str ); + TQSqlQuery elementToLoad( command, database ); // Run the query + + int id = -1; + if ( elementToLoad.isActive() && elementToLoad.first() ) + id = elementToLoad.value( 0 ).toInt(); + + return id; +} + +int TQSqlRecipeDB::findExistingRecipeByName( const TQString& name ) +{ + TQString search_str = escapeAndEncode( name.left( maxRecipeTitleLength() ) ); //truncate to the maximum size db holds + + TQString command = TQString( "SELECT id FROM recipes WHERE title LIKE '%1';" ).arg( search_str ); + TQSqlQuery elementToLoad( command, database ); // Run the query + + int id = -1; + if ( elementToLoad.isActive() && elementToLoad.first() ) + id = elementToLoad.value( 0 ).toInt(); + + return id; +} + +int TQSqlRecipeDB::findExistingYieldTypeByName( const TQString& name ) +{ + TQString search_str = escapeAndEncode( name.left( maxYieldTypeLength() ) ); //truncate to the maximum size db holds + + TQString command = TQString( "SELECT id FROM yield_types WHERE name LIKE '%1';" ).arg( search_str ); + TQSqlQuery elementToLoad( command, database ); // Run the query + + int id = -1; + if ( elementToLoad.isActive() && elementToLoad.first() ) + id = elementToLoad.value( 0 ).toInt(); + + return id; +} + +void TQSqlRecipeDB::mergeAuthors( int id1, int id2 ) +{ + TQSqlQuery update( TQString::null, database ); + + //change all instances of 'id2' to 'id1' + TQString command = TQString( "UPDATE author_list SET author_id=%1 WHERE author_id=%2" ) + .arg( id1 ) + .arg( id2 ); + update.exec( command ); + + //and ensure no duplicates were created in this process + command = TQString( "SELECT recipe_id FROM author_list WHERE author_id=%1 ORDER BY recipe_id" ) + .arg( id1 ); + update.exec( command ); + int last_id = -1; + if ( update.isActive() ) { + while ( update.next() ) { + int current_id = update.value( 0 ).toInt(); + if ( last_id == current_id ) { + int count = -1; + command = TQString( "SELECT COUNT(1) FROM author_list WHERE author_id=%1 AND recipe_id=%2" ) + .arg( id1 ) + .arg( last_id ); + TQSqlQuery remove( command, database); + if ( remove.isActive() && remove.first() ) + count = remove.value(0).toInt(); + if ( count > 1 ) { + command = TQString( "DELETE FROM author_list WHERE author_id=%1 AND recipe_id=%2" ) + .arg( id1 ) + .arg( last_id ); + database->exec( command ); + + command = TQString( "INSERT INTO author_list VALUES(%1,%2)" ) + .arg( id1 ) + .arg( last_id ); + database->exec( command ); + } + } + last_id = current_id; + } + } + + //remove author with id 'id2' + command = TQString( "DELETE FROM authors WHERE id=%1" ).arg( id2 ); + update.exec( command ); + emit authorRemoved( id2 ); +} + +void TQSqlRecipeDB::mergeCategories( int id1, int id2 ) +{ + TQSqlQuery update( TQString::null, database ); + + //change all instances of 'id2' to 'id1' + TQString command = TQString( "UPDATE category_list SET category_id=%1 WHERE category_id=%2" ) + .arg( id1 ) + .arg( id2 ); + update.exec( command ); + + //and ensure no duplicates were created in this process + command = TQString( "SELECT recipe_id FROM category_list WHERE category_id=%1 ORDER BY recipe_id" ) + .arg( id1 ); + update.exec( command ); + int last_id = -1; + if ( update.isActive() ) { + while ( update.next() ) { + int current_id = update.value( 0 ).toInt(); + if ( last_id == current_id ) { + int count = -1; + command = TQString( "SELECT COUNT(1) FROM category_list WHERE category_id=%1 AND recipe_id=%2" ) + .arg( id1 ) + .arg( last_id ); + TQSqlQuery remove( command, database); + if ( remove.isActive() && remove.first() ) + count = remove.value(0).toInt(); + if ( count > 1 ) { + command = TQString( "DELETE FROM category_list WHERE category_id=%1 AND recipe_id=%2" ) + .arg( id1 ) + .arg( last_id ); + database->exec( command ); + + command = TQString( "INSERT INTO category_list VALUES(%1,%2)" ) + .arg( id1 ) + .arg( last_id ); + database->exec( command ); + } + } + last_id = current_id; + } + } + + command = TQString( "UPDATE categories SET parent_id=%1 WHERE parent_id=%2" ) + .arg( id1 ) + .arg( id2 ); + update.exec( command ); + + //we don't want to have a category be its own parent... + command = TQString( "UPDATE categories SET parent_id=-1 WHERE parent_id=id" ); + update.exec( command ); + + //remove category with id 'id2' + command = TQString( "DELETE FROM categories WHERE id=%1" ).arg( id2 ); + update.exec( command ); + + emit categoriesMerged( id1, id2 ); +} + +void TQSqlRecipeDB::mergeIngredientGroups( int id1, int id2 ) +{ + TQSqlQuery update( TQString::null, database ); + + //change all instances of 'id2' to 'id1' + TQString command = TQString( "UPDATE ingredient_list SET group_id=%1 WHERE group_id=%2" ) + .arg( id1 ) + .arg( id2 ); + update.exec( command ); + + //remove ingredient with id 'id2' + command = TQString( "DELETE FROM ingredient_groups WHERE id=%1" ).arg( id2 ); + update.exec( command ); + emit ingGroupRemoved( id2 ); +} + +void TQSqlRecipeDB::mergeIngredients( int id1, int id2 ) +{ + TQSqlQuery update( TQString::null, database ); + + //change all instances of 'id2' to 'id1' + TQString command = TQString( "UPDATE ingredient_list SET ingredient_id=%1 WHERE ingredient_id=%2" ) + .arg( id1 ) + .arg( id2 ); + update.exec( command ); + + //delete nutrient info associated with ingredient with id 'id2' + command = TQString( "DELETE FROM ingredient_info WHERE ingredient_id=%1" ) + .arg( id2 ); + update.exec( command ); + + //update the unit_list + command = TQString( "UPDATE unit_list SET ingredient_id=%1 WHERE ingredient_id=%2" ) + .arg( id1 ) + .arg( id2 ); + update.exec( command ); + + //and ensure no duplicates were created in this process + command = TQString( "SELECT unit_id FROM unit_list WHERE ingredient_id=%1 ORDER BY unit_id" ) + .arg( id1 ); + update.exec( command ); + int last_id = -1; + if ( update.isActive() ) { + while ( update.next() ) { + int current_id = update.value( 0 ).toInt(); + if ( last_id == current_id ) { + int count = -1; + command = TQString( "SELECT COUNT(1) FROM unit_list WHERE ingredient_id=%1 AND unit_id=%2" ) + .arg( id1 ) + .arg( last_id ); + TQSqlQuery remove( command, database); + if ( remove.isActive() && remove.first() ) + count = remove.value(0).toInt(); + if ( count > 1 ) { + command = TQString( "DELETE FROM unit_list WHERE ingredient_id=%1 AND unit_id=%2" ) + .arg( id1 ) + .arg( last_id ); + database->exec( command ); + + command = TQString( "INSERT INTO unit_list VALUES(%1,%2)" ) + .arg( id1 ) + .arg( last_id ); + database->exec( command ); + } + } + last_id = current_id; + } + } + + //update ingredient info + command = TQString( "UPDATE ingredient_info SET ingredient_id=%1 WHERE ingredient_id=%2" ) + .arg( id1 ) + .arg( id2 ); + update.exec( command ); + + //and ensure no duplicates were created in this process + //info associated with one ingredient will be lost... they should be the same ingredient and thus info anyways + command = TQString( "SELECT property_id FROM ingredient_info WHERE ingredient_id=%1 ORDER BY property_id" ) + .arg( id1 ); + update.exec( command ); + last_id = -1; + if ( update.isActive() ) { + while ( update.next() ) { + int current_id = update.value( 0 ).toInt(); + if ( last_id == current_id ) { + int count = -1; + command = TQString( "SELECT COUNT(1) FROM ingredient_info WHERE ingredient_id=%1 AND property_id=%2" ) + .arg( id1 ) + .arg( last_id ); + TQSqlQuery remove( command, database); + if ( remove.isActive() && remove.first() ) + count = remove.value(0).toInt(); + if ( count > 1 ) { + command = TQString( "DELETE FROM ingredient_info WHERE ingredient_id=%1 AND property_id=%2" ) + .arg( id1 ) + .arg( last_id ); + database->exec( command ); + + command = TQString( "INSERT INTO ingredient_info VALUES(%1,%2)" ) + .arg( id1 ) + .arg( last_id ); + database->exec( command ); + } + + } + last_id = current_id; + } + } + + //remove ingredient with id 'id2' + command = TQString( "DELETE FROM ingredients WHERE id=%1" ).arg( id2 ); + update.exec( command ); + emit ingredientRemoved( id2 ); +} + +void TQSqlRecipeDB::mergePrepMethods( int id1, int id2 ) +{ + TQSqlQuery update( TQString::null, database ); + + //change all instances of 'id2' to 'id1' in ingredient list + TQString command = TQString( "UPDATE prep_method_list SET prep_method_id=%1 WHERE prep_method_id=%2" ) + .arg( id1 ) + .arg( id2 ); + update.exec( command ); + + //change all instances of 'id2' to 'id1' in ingredient weights + command = TQString( "UPDATE ingredient_weights SET prep_method_id=%1 WHERE prep_method_id=%2" ) + .arg( id1 ) + .arg( id2 ); + update.exec( command ); + + //remove prep method with id 'id2' + command = TQString( "DELETE FROM prep_methods WHERE id=%1" ).arg( id2 ); + update.exec( command ); + emit prepMethodRemoved( id2 ); +} + +void TQSqlRecipeDB::mergeProperties( int id1, int id2 ) +{ + TQSqlQuery update( TQString::null, database ); + + //change all instances of 'id2' to 'id1' + TQString command = TQString( "UPDATE ingredient_properties SET id=%1 WHERE id=%2" ) + .arg( id1 ) + .arg( id2 ); + update.exec( command ); + + command = TQString( "UPDATE ingredient_info SET property_id=%1 WHERE property_id=%2" ) + .arg( id1 ) + .arg( id2 ); + update.exec( command ); + + //remove prep method with id 'id2' + command = TQString( "DELETE FROM ingredient_properties WHERE id=%1" ).arg( id2 ); + update.exec( command ); + emit propertyRemoved( id2 ); +} + +void TQSqlRecipeDB::mergeUnits( int id1, int id2 ) +{ + TQSqlQuery update( TQString::null, database ); + + //change all instances of 'id2' to 'id1' in unit list + TQString command = TQString( "UPDATE unit_list SET unit_id=%1 WHERE unit_id=%2" ) + .arg( id1 ) + .arg( id2 ); + update.exec( command ); + + //change all instances of 'id2' to 'id1' in ingredient list + command = TQString( "UPDATE ingredient_list SET unit_id=%1 WHERE unit_id=%2" ) + .arg( id1 ) + .arg( id2 ); + update.exec( command ); + + //and ensure no duplicates were created in this process + command = TQString( "SELECT ingredient_id FROM unit_list WHERE unit_id=%1 ORDER BY ingredient_id" ) + .arg( id1 ); + update.exec( command ); + int last_id = -1; + if ( update.isActive() ) { + while ( update.next() ) { + int current_id = update.value( 0 ).toInt(); + if ( last_id == current_id ) { + int count = -1; + command = TQString( "SELECT COUNT(1) FROM unit_list WHERE ingredient_id=%1 AND unit_id=%2" ) + .arg( id1 ) + .arg( last_id ); + TQSqlQuery remove( command, database); + if ( remove.isActive() && remove.first() ) + count = remove.value(0).toInt(); + if ( count > 1 ) { + command = TQString( "DELETE FROM unit_list WHERE ingredient_id=%1 AND unit_id=%2" ) + .arg( id1 ) + .arg( last_id ); + database->exec( command ); + + command = TQString( "INSERT INTO unit_list VALUES(%1,%2)" ) + .arg( id1 ) + .arg( last_id ); + database->exec( command ); + } + } + last_id = current_id; + } + } + + //update ingredient info + command = TQString( "UPDATE ingredient_info SET per_units=%1 WHERE per_units=%2" ) + .arg( id1 ) + .arg( id2 ); + update.exec( command ); + + //change all instances of 'id2' to 'id1' in unit_conversion + command = TQString( "UPDATE units_conversion SET unit1_id=%1 WHERE unit1_id=%2" ) + .arg( id1 ) + .arg( id2 ); + update.exec( command ); + command = TQString( "UPDATE units_conversion SET unit2_id=%1 WHERE unit2_id=%2" ) + .arg( id1 ) + .arg( id2 ); + update.exec( command ); + + //and ensure that the one to one ratio wasn't created + command = TQString( "DELETE FROM units_conversion WHERE unit1_id=unit2_id" ); + update.exec( command ); + + //update ingredient weights + command = TQString( "UPDATE ingredient_weights SET unit_id=%1 WHERE unit_id=%2" ) + .arg( id1 ) + .arg( id2 ); + update.exec( command ); + command = TQString( "UPDATE ingredient_weights SET weight_unit_id=%1 WHERE weight_unit_id=%2" ) + .arg( id1 ) + .arg( id2 ); + update.exec( command ); + + //change all instances of 'id2' to 'id1' in ingredient weights + command = TQString( "UPDATE ingredient_weights SET unit_id=%1 WHERE unit_id=%2" ) + .arg( id1 ) + .arg( id2 ); + update.exec( command ); + + command = TQString( "UPDATE ingredient_weights SET weight_unit_id=%1 WHERE weight_unit_id=%2" ) + .arg( id1 ) + .arg( id2 ); + update.exec( command ); + + //remove units with id 'id2' + command = TQString( "DELETE FROM units WHERE id=%1" ).arg( id2 ); + update.exec( command ); + emit unitRemoved( id2 ); +} + +TQString TQSqlRecipeDB::getUniqueRecipeTitle( const TQString &recipe_title ) +{ + //already is unique + if ( findExistingRecipeByName( recipe_title ) == -1 ) + return recipe_title; + + TQString return_title = recipe_title; //If any error is produced, just go for default value (always return something) + + TQString command = TQString( "SELECT COUNT(1) FROM recipes WHERE title LIKE '%1 (%)';" ).arg( escapeAndEncode( recipe_title ) ); + + TQSqlQuery alikeRecipes( command, database ); + if ( alikeRecipes.isActive() && alikeRecipes.first() ) + { + int count = alikeRecipes.value( 0 ).toInt(); + return_title = TQString( "%1 (%2)" ).arg( recipe_title ).arg( count + 2 ); + + //make sure this newly created title is unique (just in case) + while ( findExistingRecipeByName( return_title ) != -1 ) { + count--; //go down to find the skipped recipe(s) + return_title = TQString( "%1 (%2)" ).arg( recipe_title ).arg( count + 2 ); + } + } + + return return_title; +} + +TQString TQSqlRecipeDB::recipeTitle( int recipeID ) +{ + TQString command = TQString( "SELECT title FROM recipes WHERE id=%1;" ).arg( recipeID ); + TQSqlQuery recipeToLoad( command, database ); + if ( recipeToLoad.isActive() && recipeToLoad.next() ) // Go to the first record (there should be only one anyway. + return ( unescapeAndDecode(recipeToLoad.value( 0 ).toCString()) ); + + return ( TQString::null ); +} + +void TQSqlRecipeDB::emptyData( void ) +{ + TQStringList tables; + tables << "ingredient_info" << "ingredient_list" << "ingredient_properties" << "ingredients" << "recipes" << "unit_list" << "units" << "units_conversion" << "categories" << "category_list" << "authors" << "author_list" << "prep_methods" << "ingredient_groups" << "yield_types" << "ratings" << "rating_criteria" << "rating_criterion_list"; + TQSqlQuery tablesToEmpty( TQString::null, database ); + for ( TQStringList::Iterator it = tables.begin(); it != tables.end(); ++it ) { + TQString command = TQString( "DELETE FROM %1;" ).arg( *it ); + tablesToEmpty.exec( command ); + } +} + +void TQSqlRecipeDB::empty( void ) +{ + TQSqlQuery tablesToEmpty( TQString::null, database ); + + TQStringList list = database->tables(); + TQStringList::const_iterator it = list.begin(); + while( it != list.end() ) { + TQString command = TQString( "DROP TABLE %1;" ).arg( *it ); + tablesToEmpty.exec( command ); + + if ( !tablesToEmpty.isActive() ) + kdDebug()<<tablesToEmpty.lastError().databaseText()<<endl; + + ++it; + } +} + +TQString TQSqlRecipeDB::getNextInsertIDStr( const TQString &table, const TQString &column ) +{ + int next_id = getNextInsertID( table, column ); + + TQString id_str; + if ( next_id == -1 ) + id_str = "NULL"; + else + id_str = TQString::number( next_id ); + + return id_str; +} + +void TQSqlRecipeDB::search( RecipeList *list, int items, const RecipeSearchParameters ¶meters ) +{ + TQString query = buildSearchQuery(parameters); + + TQValueList<int> ids; + TQSqlQuery recipeToLoad( query, database ); + if ( recipeToLoad.isActive() ) { + while ( recipeToLoad.next() ) { + ids << recipeToLoad.value( 0 ).toInt(); + } + } + + if ( ids.count() > 0 ) + loadRecipes( list, items, ids ); +} + +#include "qsqlrecipedb.moc" diff --git a/src/backends/qsqlrecipedb.h b/src/backends/qsqlrecipedb.h new file mode 100644 index 0000000..b9211d7 --- /dev/null +++ b/src/backends/qsqlrecipedb.h @@ -0,0 +1,228 @@ +/*************************************************************************** +* Copyright (C) 2004 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* Copyright (C) 2006 Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + + +#ifndef TQSQLRECIPEDB_H +#define TQSQLRECIPEDB_H + +#include "backends/recipedb.h" + +#include <tqglobal.h> +#include <tqobject.h> +#include <tqsqldatabase.h> +#include <tqimage.h> +#include <tqfileinfo.h> +#include <tqregexp.h> +#include <tqstring.h> + +#include "datablocks/recipe.h" +#include "datablocks/elementlist.h" +#include "datablocks/ingredientpropertylist.h" +#include "datablocks/unitratiolist.h" + +/** +@author Unai Garro, Jason Kivlighn +*/ +class TQSqlRecipeDB : public RecipeDB +{ + + TQ_OBJECT + +protected: + virtual TQString qsqlDriverPlugin() const { return TQString::null; } + virtual TQSqlDriver *qsqlDriver() const { return 0; } + virtual void createDB( void ) = 0; + + virtual void portOldDatabases( float version ); + virtual void storePhoto( int recipeID, const TQByteArray &data ); + virtual void loadPhoto( int recipeID, TQPixmap &photo ); + void loadRecipeMetadata( Recipe *recipe ); + + void search( RecipeList *list, int items, const RecipeSearchParameters & ); + + /** Return the next id for the given table and column. + * If the database supports getting this afterwards, + * leave the default implementation which returns -1. + * + * Note: Only call when an insert is actually going to take place. + * This function will increment the sequence counter. + */ + virtual int getNextInsertID( const TQString & /*table*/, const TQString & /*column*/ ) + { + return -1; + } + + TQSqlDatabase *database; + TQSqlQuery m_query; + TQString DBuser; + TQString DBpass; + TQString DBhost; + int DBport; + +public: + TQSqlRecipeDB( const TQString &host, const TQString &user = TQString::null, const TQString &pass = TQString::null, const TQString &DBName = DEFAULT_DB_NAME, int port = 0 ); + ~TQSqlRecipeDB( void ); + + void connect( bool create_db, bool create_tables ); + + void addIngredientWeight( const Weight & ); + void addProperty( const TQString &name, const TQString &units ); + void addPropertyToIngredient( int ingredientID, int propertyID, double amount, int perUnitsID ); + void addUnitToIngredient( int ingredientID, int unitID ); + + void categorizeRecipe( int recipeID, const ElementList &categoryList ); + void changePropertyAmountToIngredient( int ingredientID, int propertyID, double amount, int per_units ); + + void createNewAuthor( const TQString &authorName ); + void createNewCategory( const TQString &categoryName, int parent_id = -1 ); + void createNewIngGroup( const TQString &name ); + void createNewIngredient( const TQString &ingredientName ); + void createNewPrepMethod( const TQString &prepMethodName ); + void createNewRating( const TQString &name ); + void createNewUnit( const Unit &unit ); + void createNewYieldType( const TQString &type ); + + void emptyData( void ); + void empty( void ); + + int findExistingAuthorByName( const TQString& name ); + int findExistingCategoryByName( const TQString& name ); + int findExistingIngredientGroupByName( const TQString& name ); + int findExistingIngredientByName( const TQString& name ); + int findExistingPrepByName( const TQString& name ); + int findExistingRecipeByName( const TQString& name ); + int findExistingRatingByName( const TQString& name ); + int findExistingUnitByName( const TQString& name ); + int findExistingPropertyByName( const TQString& name ); + int findExistingYieldTypeByName( const TQString& name ); + void findIngredientUnitDependancies( int ingredientID, int unitID, ElementList *recipes, ElementList *ingredientInfo ); + void findIngredientDependancies( int ingredientID, ElementList *recipes ); + void findPrepMethodDependancies( int prepMethodID, ElementList *recipes ); + void findUnitDependancies( int unitID, ElementList *properties, ElementList *recipes, ElementList *weights ); + void findUseOfIngGroupInRecipes( ElementList *results, int groupID ); + void findUseOfCategoryInRecipes( ElementList *results, int catID ); + void findUseOfAuthorInRecipes( ElementList *results, int authorID ); + + TQString getUniqueRecipeTitle( const TQString &recipe_title ); + + bool ingredientContainsProperty( int ingredientID, int propertyID, int perUnitsID ); + bool ingredientContainsUnit( int ingredientID, int unitID ); + + void loadAuthors( ElementList *list, int limit = -1, int offset = 0 ); + void loadCategories( CategoryTree *list, int limit = -1, int offset = 0, int parent_id = -1, bool recurse = true ); + void loadCategories( ElementList *list, int limit = -1, int offset = 0 ); + void loadIngredientGroups( ElementList *list ); + void loadIngredients( ElementList *list, int limit = -1, int offset = 0 ); + void loadPossibleUnits( int ingredientID, UnitList *list ); + void loadPrepMethods( ElementList *list, int limit = -1, int offset = 0 ); + void loadProperties( IngredientPropertyList *list, int ingredientID = -2 ); // Loads the list of possible properties by default, all the ingredient properties with -1, and the ingredients of given property if id>=0 + void loadRatingCriterion( ElementList *list, int limit = -1, int offset = 0 ); + void loadRecipes( RecipeList *, int items = All, TQValueList<int> ids = TQValueList<int>() ); + void loadRecipeList( ElementList *list, int categoryID = -1, bool recursive = false ); + void loadUncategorizedRecipes( ElementList *list ); + void loadUnits( UnitList *list, Unit::Type = Unit::All, int limit = -1, int offset = 0 ); + void loadUnitRatios( UnitRatioList *ratioList, Unit::Type ); + void loadYieldTypes( ElementList *list, int limit, int offset ); + + void mergeAuthors( int id1, int id2 ); + void mergeCategories( int id1, int id2 ); + void mergeIngredientGroups( int id1, int id2 ); + void mergeIngredients( int id1, int id2 ); + void mergeUnits( int id1, int id2 ); + void mergePrepMethods( int id1, int id2 ); + void mergeProperties( int id1, int id2 ); + + void modIngredientGroup( int ingredientID, const TQString &newLabel ); + /** + * set newLabel for ingredientID + */ + void modIngredient( int ingredientID, const TQString &newLabel ); + /** + * set newLabel for unitID + */ + void modUnit( const Unit &unit ); + /** + * set newLabel for categoryID + */ + void modCategory( int categoryID, const TQString &newLabel ); + void modCategory( int categoryID, int new_parent_id ); + /** + * set newLabel for authorID + */ + void modAuthor( int authorID, const TQString &newLabel ); + + void modPrepMethod( int prepMethodID, const TQString &newLabel ); + + void modProperty( int propertyID, const TQString &newLabel ); + + TQString recipeTitle( int recipeID ); + + void removeAuthor( int categoryID ); + void removeCategory( int categoryID ); + void removeIngredientGroup( int groupID ); + void removeIngredient( int ingredientID ); + void removeIngredientWeight( int id ); + void removePrepMethod( int prepMethodID ); + void removeProperty( int propertyID ); + void removePropertyFromIngredient( int ingredientID, int propertyID, int perUnitID ); + void removeRecipe( int id ); + void removeRecipeFromCategory( int ingredientID, int categoryID ); + void removeUnit( int unitID ); + void removeUnitFromIngredient( int ingredientID, int unitID ); + void removeUnitRatio( int unitID1, int unitID2 ); + + void saveRecipe( Recipe *recipe ); + void saveUnitRatio( const UnitRatio *ratio ); + + double unitRatio( int unitID1, int unitID2 ); + double ingredientWeight( const Ingredient &ing, bool *wasApproximated = 0 ); + WeightList ingredientWeightUnits( int ingID ); + + TQString escapeAndEncode( const TQString &s ) const; + TQString unescapeAndDecode( const TQCString &s ) const; + + TQString categoryName( int ID ); + TQString prepMethodName( int ID ); + TQString ingredientName( int ID ); + IngredientProperty propertyName( int ID ); + Unit unitName( int ID ); + + int getCount( const TQString &table_name ); + int categoryTopLevelCount(); + + bool checkIntegrity( void ); + + void splitCommands( TQString& s, TQStringList& sl ); + + float databaseVersion( void ); + +protected: + void execSQL( const TQString &command ); + +private: + void loadElementList( ElementList *elList, TQSqlQuery *query ); + void loadPropertyElementList( ElementList *elList, TQSqlQuery *query ); + TQString getNextInsertIDStr( const TQString &table, const TQString &column ); + + TQString DBname; + const TQString connectionName; + TQString m_command; + + static int m_refCount; +}; + + + + +#endif diff --git a/src/backends/recipedb.cpp b/src/backends/recipedb.cpp new file mode 100644 index 0000000..fbc0ff4 --- /dev/null +++ b/src/backends/recipedb.cpp @@ -0,0 +1,989 @@ +/*************************************************************************** +* Copyright (C) 2003 * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* * +* Copyright (C) 2003-2006 Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "backends/recipedb.h" + +#ifdef HAVE_CONFIG_H +#include "config.h" +#endif + +#include <tdeapplication.h> +#include <tdeconfig.h> +#include <kdebug.h> +#include <kstandarddirs.h> +#include <kprogress.h> +#include <tdeglobal.h> +#include <tdelocale.h> +#include <tdeaboutdata.h> +#include <kprocess.h> +#include <kprocio.h> +#include <kfilterdev.h> +#include <tdemessagebox.h> + +#include <tqfile.h> +#include <tqstringlist.h> +#include <tqtextstream.h> + +#include <map> + +#include "importers/kreimporter.h" + +#if HAVE_POSTGRESQL +#include "PostgreSQL/psqlrecipedb.h" +#endif + +#if HAVE_MYSQL +#include "MySQL/mysqlrecipedb.h" +#endif + +#if HAVE_SQLITE || HAVE_SQLITE3 +#include "SQLite/literecipedb.h" +#endif + +#include "datablocks/categorytree.h" +#include "datablocks/ingredientpropertylist.h" +#include "datablocks/weight.h" + +#include "searchparameters.h" + +#include "usda_property_data.h" +#include "usda_ingredient_data.h" +#include "usda_unit_data.h" + +#define DB_FILENAME "krecipes.krecdb" + +struct ingredient_nutrient_data +{ + int usda_id; + TQString name; + TQValueList<double> data; + WeightList weights; +}; + +RecipeDB::RecipeDB() : + DCOPObject(), + TQObject(), m_categoryCache(0), haltOperation(false) +{ + dbOK = false; + dbErr = ""; +} + +RecipeDB::~RecipeDB() +{ +} + +double RecipeDB::latestDBVersion() const +{ + return 0.95; +} + +TQString RecipeDB::krecipes_version() const +{ + TDEInstance * this_instance = TDEGlobal::instance(); + if ( this_instance && this_instance->aboutData() ) + return this_instance->aboutData() ->version(); + + return TQString::null; //Oh, well. We couldn't get the version (shouldn't happen). +} + +RecipeDB* RecipeDB::createDatabase( const TQString &dbType, const TQString &file ) +{ + TDEConfig * config = kapp->config(); + config->setGroup( "Server" ); + TQString host = config->readEntry( "Host", "localhost" ); + TQString user = config->readEntry( "Username", TQString::null ); + TQString pass = config->readEntry( "Password", TQString::null ); + TQString dbname = config->readEntry( "DBName", DEFAULT_DB_NAME ); + int port = config->readNumEntry( "Port", 0 ); + + TQString f = file; + if ( f.isEmpty() ) + f = config->readEntry( "DBFile", locateLocal ( "appdata", DB_FILENAME ) ); + + return createDatabase( dbType, host, user, pass, dbname, port, f ); +} + +RecipeDB* RecipeDB::createDatabase( const TQString &dbType, const TQString &host, const TQString &user, const TQString &pass, const TQString &dbname, int port, const TQString &file ) +{ + RecipeDB * database = 0; + + if ( 0 ) + ; //we need some condition here +#if HAVE_SQLITE || HAVE_SQLITE3 + + else if ( dbType == "SQLite" ) { + database = new LiteRecipeDB( file ); + } +#endif //HAVE_SQLITE || HAVE_SQLITE3 + #if HAVE_MYSQL + else if ( dbType == "MySQL" ) { + database = new MySQLRecipeDB( host, user, pass, dbname, port ); + } +#endif //HAVE_MYSQL + #if HAVE_POSTGRESQL + else if ( dbType == "PostgreSQL" ) { + database = new PSqlRecipeDB( host, user, pass, dbname, port ); + } +#endif //HAVE_POSTGRESQL + else { + kdDebug() << "No database support included (or available) for the " << dbType << " database." << endl; + } + + return database; +} + +void RecipeDB::updateCategoryCache( int limit ) +{ + m_categoryCache = new CategoryTree; + loadCategories( m_categoryCache, limit, 0, -1, true ); +} + +void RecipeDB::clearCategoryCache() +{ + delete m_categoryCache; + m_categoryCache = 0; +} + +void RecipeDB::loadCachedCategories( CategoryTree **list, int limit, int offset, int parent_id, bool recurse ) +{ + if ( m_categoryCache ) { + if ( parent_id == -1 ) + *list = m_categoryCache; + else //FIXME?: how slow is this find() call? the cache is loaded in sequential order, so should we iterate over the cache? + *list = m_categoryCache->find(parent_id); + //kdDebug() << "Loading category tree from the cache" << endl; + } + else { + loadCategories( *list, limit, offset, parent_id, recurse ); + } +} + +RecipeDB::ConversionStatus RecipeDB::convertIngredientUnits( const Ingredient &from, const Unit &to, Ingredient &result ) +{ + result = from; + + if ( from.units.id == to.id ) + return Success; + + if ( from.units.type == to.type && to.type != Unit::Other ) { + double ratio = unitRatio( from.units.id, to.id ); + if ( ratio > 0 ) { + result.amount = from.amount * ratio; + result.units = to; + + kdDebug()<<"Unit conversion SUCCESSFUL, from "<<unitName(from.units.id).name<<" to "<<unitName(to.id).name<<" for ingredient "<<ingredientName(from.ingredientID)<<" (ingredient not used in conversion)"<<endl; + + return Success; + } + else { + kdDebug()<<"Unit conversion failed, you should probably update your unit conversion table."<<endl; + kdDebug()<<from.units.id<<" to "<<to.id<<endl; + return MissingUnitConversion; + } + } + else if ( to.type == Unit::Mass || from.units.type == Unit::Mass ) { + if ( from.ingredientID == -1 ) + return MissingIngredient; + + double fromToWeightRatio, weightToToRatio; + int unitID = -1; + int prepID = -2; + + WeightList idList = ingredientWeightUnits( from.ingredientID ); + + if ( idList.count() == 0 ) + return MissingIngredientWeight; + + for ( WeightList::const_iterator it = idList.begin(); it != idList.end(); ++it ) { + //get conversion order correct (i.e., Mass -> Volume instead of Volume -> Mass, depending on unit type) + int first = (to.type == Unit::Mass)?(*it).perAmountUnitID:(*it).weightUnitID; + int second = (to.type == Unit::Mass)?(*it).weightUnitID:(*it).perAmountUnitID; + double tryFromToWeightRatio = unitRatio( from.units.id, first ); + if ( tryFromToWeightRatio > 0 ) { + weightToToRatio = unitRatio( second, to.id ); + fromToWeightRatio = tryFromToWeightRatio; + unitID = first; + + kdDebug()<<"units work, is it the right prep method..."<<endl; + if ( from.prepMethodList.containsId( (*it).prepMethodID ) ) { + kdDebug()<<" yes"<<endl; + prepID = (*it).prepMethodID; + break; + } + kdDebug()<<" no, keep going"<<endl; + } + } + if ( unitID == -1 ) + return MissingUnitConversion; + + bool wasApproximated; + + Ingredient i; + i.ingredientID = from.ingredientID; + i.units.id = unitID; + i.amount = from.amount * fromToWeightRatio; + i.prepMethodList = from.prepMethodList; + result.amount = ingredientWeight( i, &wasApproximated ) * weightToToRatio; + result.units = to; + + if ( result.amount < 0 ) + return MismatchedPrepMethod; + else if ( wasApproximated ) + return MismatchedPrepMethodUsingApprox; + + return Success; + } + else { + TQString to_str; + switch ( to.type ) { + case Unit::Other: to_str = "Other"; break; + case Unit::Mass: to_str = "Mass"; break; + case Unit::Volume: to_str = "Volume"; break; + case Unit::All: kdDebug()<<"Code error: trying to convert to unit of type 'All'"<<endl; return InvalidTypes; + } + TQString from_str; + switch ( from.units.type ) { + case Unit::Other: from_str = "Other"; break; + case Unit::Mass: from_str = "Mass"; break; + case Unit::Volume: from_str = "Volume"; break; + case Unit::All: kdDebug()<<"Code error: trying to convert from unit of type 'All'"<<endl; return InvalidTypes; + } + kdDebug()<<"Can't handle conversion from "<<from_str<<"("<<from.units.id<<") to "<<to_str<<"("<<to.id<<")"<<endl; + + return InvalidTypes; + } +} + +bool RecipeDB::backup( const TQString &backup_file, TQString *errMsg ) +{ + kdDebug()<<"Backing up current database to "<<backup_file<<endl; + + TDEProcess *p = new TDEProcess; + //p->setUseShell(true); + + TQIODevice *dumpFile = KFilterDev::deviceForFile(backup_file,"application/x-gzip"); + if ( !dumpFile->open( IO_WriteOnly ) ) { + kdDebug()<<"Couldn't open "<<backup_file<<endl; + return false; + } + + dumpStream = new TQTextStream( dumpFile ); + + TQStringList command = backupCommand(); + if ( command.count() == 0 ) { + kdDebug()<<"Backup not available for this database backend"<<endl; + return false; + } + + TDEConfig * config = kapp->config(); + config->setGroup( "DBType" ); + + (*dumpStream) << "-- Generated for Krecipes v"<<krecipes_version()<<endl; + (*dumpStream) << "-- Krecipes database schema: "<<latestDBVersion()<<endl; + (*dumpStream) << "-- Krecipes database backend: "<<config->readEntry( "Type" )<<endl; + + kdDebug()<<"Running '"<<command.first()<<"' to create backup file"<<endl; + *p << command /*<< ">" << backup_file*/; + + TQApplication::connect( p, TQ_SIGNAL(receivedStdout(TDEProcess*,char*,int)), this, TQ_SLOT(processDumpOutput(TDEProcess*,char*,int)) ); + TQApplication::connect( p, TQ_SIGNAL(receivedStderr(TDEProcess*,char*,int)), this, TQ_SLOT(processDumpOutput(TDEProcess*,char*,int)) ); + + emit progressBegin(0,TQString::null, + TQString("<center><b>%1</b></center>%2") + .arg(i18n("Creating complete backup")) + .arg(i18n("Depending on the number of recipes and amount of data, this could take some time.")),50); + + bool success = p->start( TDEProcess::Block, TDEProcess::AllOutput ); + if ( !success ) { + if ( errMsg ) *errMsg = TQString(i18n("Unable to find or run the program '%1'. Either it is not installed on your system or it is not in $PATH.")).arg(command.first()); + delete p; + delete dumpStream; + delete dumpFile; + TQFile::remove(backup_file); + emit progressDone(); + return false; + } + + emit progressDone(); + + //User cancelled it; we'll still consider the operation successful, + //but delete the file we created + if ( !p->normalExit() ) { + kdDebug()<<"Process killed, deleting partial backup."<<endl; + TQFile::remove(backup_file); + } + + if ( p->exitStatus() != 0 ) { + //Since the process failed, dumpStream should have output from the app as to why it did + TQString appOutput; + dumpFile->close(); + if ( dumpFile->open( IO_ReadOnly ) ) { + TQTextStream appErrStream( dumpFile ); + + //ignore our own versioning output + appErrStream.readLine(); + appErrStream.readLine(); + appErrStream.readLine(); + + appOutput = appErrStream.read(); + } + else + kdDebug()<<"Unable to open file to get error output."<<endl; + + if ( errMsg ) *errMsg = TQString("%1\n%2").arg(i18n("Backup failed.")).arg(appOutput); + TQFile::remove(backup_file); + delete p; + delete dumpStream; + delete dumpFile; + return false; + } + + delete p; + delete dumpStream; + delete dumpFile; + return true; +} + +void RecipeDB::processDumpOutput( TDEProcess *p, char *buffer, int buflen ) +{ + int written = dumpStream->device()->writeBlock(buffer,buflen); + if ( written != buflen ) + kdDebug()<<"Data lost: written ("<<written<<") != buflen ("<<buflen<<")"<<endl; + + if ( haltOperation ) { haltOperation=false; p->kill(); return; } + emit progress(); +} + +void RecipeDB::initializeData( void ) +{ + // Populate with data + + // Read the commands form the data file + TQFile datafile( TDEGlobal::dirs() ->findResource( "appdata", "data/data.sql" ) ); + if ( datafile.open( IO_ReadOnly ) ) { + TQTextStream stream( &datafile ); + execSQL(stream); + datafile.close(); + } +} + +bool RecipeDB::restore( const TQString &file, TQString *errMsg ) +{ + TQIODevice *dumpFile = KFilterDev::deviceForFile(file,"application/x-gzip"); + if ( dumpFile->open( IO_ReadOnly ) ) { + + TQTextStream stream( dumpFile ); + TQString firstLine = stream.readLine().stripWhiteSpace(); + TQString dbVersion = stream.readLine().stripWhiteSpace(); + dbVersion = dbVersion.right( dbVersion.length() - dbVersion.find(":") - 2 ); + if ( tqRound(dbVersion.toDouble()*1e5) > tqRound(latestDBVersion()*1e5) ) { //correct for float's imprecision + if ( errMsg ) *errMsg = i18n( "This backup was created with a newer version of Krecipes and cannot be restored." ); + delete dumpFile; + return false; + } + + TDEConfig * config = kapp->config(); + config->setGroup( "DBType" ); + TQString dbType = stream.readLine().stripWhiteSpace(); + dbType = dbType.right( dbType.length() - dbType.find(":") - 2 ); + if ( dbType.isEmpty() || !firstLine.startsWith("-- Generated for Krecipes") ) { + if ( errMsg ) *errMsg = i18n("This file is not a Krecipes backup file or has become corrupt."); + delete dumpFile; + return false; + } + else if ( dbType != config->readEntry("Type",TQString::null) ) { + if ( errMsg ) *errMsg = TQString(i18n("This backup was created using the \"%1\" backend. It can only be restored into a database using this backend." )).arg(dbType); + delete dumpFile; + return false; + } + + + //We have to first wipe the database structure. Note that if we load a dump + //with from a previous version of Krecipes, the difference in structure + // wouldn't allow the data to be inserted. This remains forward-compatibity + //by loading the old schema and then porting it to the current version. + empty(); //the user had better be warned! + + KProcIO *process = new KProcIO; + + TQStringList command = restoreCommand(); + kdDebug()<<"Restoring backup using: "<<command[0]<<endl; + *process << command; + + //process->setComm( TDEProcess::Stdin ); + if ( process->start( TDEProcess::NotifyOnExit ) ) { + emit progressBegin(0,TQString::null, + TQString("<center><b>%1</b></center>%2") + .arg(i18n("Restoring backup")) + .arg(i18n("Depending on the number of recipes and amount of data, this could take some time."))); + + do { + TQByteArray array(4096); + int len = dumpFile->readBlock(array.data(),array.size()); + array.resize(len); + + if ( !process->writeStdin(array) ) + kdDebug()<<"Yikes! Some input couldn't be written to the process!"<<endl; + + if ( haltOperation ) { break; } + emit progress(); + } + while ( !stream.atEnd() ); + + process->closeWhenDone(); + + //Since the process will exit when all stdin has been sent and processed, + //just loop until the process is no longer running. If something goes + //wrong, the user can still hit cancel. + int prog = 0; + while ( process->isRunning() ){ + if ( haltOperation ) { break; } + kapp->processEvents(); + if ( prog % 100 == 0 ) { + emit progress(); + prog = 0; + } + ++prog; + } + } + else + kdDebug()<<"Unable to start process"<<endl; + + delete process; + emit progressDone(); + + //Since we just loaded part of a file, the database won't be in a usable state. + //We'll wipe out the database structure and recreate it, leaving no data. + if ( haltOperation ) { + haltOperation=false; + empty(); + checkIntegrity(); + delete dumpFile; + if ( errMsg ) { *errMsg = i18n("Restore Failed"); } + return false; + } + + dumpFile->close(); + + checkIntegrity(); + } + else { + kdDebug()<<"Unable to open the selected backup file"<<endl; + return false; + } + + delete dumpFile; + return true; +} + +void RecipeDB::execSQL( TQTextStream &stream ) +{ + TQString line, command; + while ( (line = stream.readLine()) != TQString::null ) { + command += " "+line; + if ( command.startsWith(" --") ) { + command = TQString::null; + } + else if ( command.endsWith(";") ) { + execSQL( command ); + command = TQString::null; + } + } +} + +void RecipeDB::loadRecipe( Recipe *recipe, int items, int id ) +{ + RecipeList rlist; + TQValueList<int> ids; ids << id; + loadRecipes( &rlist, items, ids ); + + *recipe = *rlist.begin(); +} + +int RecipeDB::categoryCount() +{ + return getCount("categories"); +} + +int RecipeDB::authorCount() +{ + return getCount("authors"); +} + +int RecipeDB::ingredientCount() +{ + return getCount("ingredients"); +} + +int RecipeDB::prepMethodCount() +{ + return getCount("prep_methods"); +} + +int RecipeDB::unitCount() +{ + return getCount("units"); +} + +void RecipeDB::importSamples() +{ + TQString sample_recipes = locate( "appdata", "data/samples-" + TDEGlobal::locale() ->language() + ".kreml" ); + if ( sample_recipes.isEmpty() ) { + //TODO: Make this a KMessageBox?? + kdDebug() << "NOTICE: Samples recipes for the language \"" << TDEGlobal::locale() ->language() << "\" are not available. However, if you would like samples recipes for this language included in future releases of Krecipes, we invite you to submit your own. Just save your favorite recipes in the kreml format and e-mail them to [email protected]." << endl; + + sample_recipes = locate( "appdata", "data/samples-en_US.kreml" ); //default to English + } + if ( !sample_recipes.isEmpty() ) { + KreImporter importer; + + TQStringList file; + file << sample_recipes; + importer.parseFiles( file ); + + importer.import( this, true ); + } + else + kdDebug() << "Unable to find samples recipe file (samples-en_US.kreml)" << endl; +} + +void RecipeDB::getIDList( const CategoryTree *categoryTree, TQStringList &ids ) +{ + for ( CategoryTree * child_it = categoryTree->firstChild(); child_it; child_it = child_it->nextSibling() ) { + ids << TQString::number(child_it->category.id); + getIDList(child_it,ids ); + } +} + +TQString RecipeDB::buildSearchQuery( const RecipeSearchParameters &p ) const +{ + TQStringList queryList, conditionList, tableList; + + if ( p.ingsOr.count() != 0 ) { + tableList << "ingredient_list il" << "ingredients i"; + conditionList << "il.ingredient_id=i.id" << "il.recipe_id=r.id"; + + TQString condition = "("; + for ( TQStringList::const_iterator it = p.ingsOr.begin(); it != p.ingsOr.end();) { + condition += "i.name LIKE '%"+escapeAndEncode(*it)+"%' "; + if ( ++it != p.ingsOr.end() ) { + condition += "OR "; + } + } + condition += ")"; + + conditionList << condition; + } + + if ( p.catsOr.count() != 0 ) { + tableList << "category_list cl" << "categories c"; + conditionList << "cl.category_id=c.id" << "cl.recipe_id=r.id"; + + TQString condition = "("; + for ( TQStringList::const_iterator it = p.catsOr.begin(); it != p.catsOr.end();) { + condition += "c.name LIKE '%"+escapeAndEncode(*it)+"%' "; + if ( ++it != p.catsOr.end() ) { + condition += "OR "; + } + } + condition += ")"; + + conditionList << condition; + } + + if ( p.authorsOr.count() != 0 ) { + tableList << "author_list al" << "authors a"; + conditionList << "al.author_id=a.id" << "al.recipe_id=r.id"; + + TQString condition = "("; + for ( TQStringList::const_iterator it = p.authorsOr.begin(); it != p.authorsOr.end();) { + condition += "a.name LIKE '%"+escapeAndEncode(*it)+"%'"; + if ( ++it != p.authorsOr.end() ) { + condition += "OR "; + } + } + condition += ")"; + + conditionList << condition; + } + + if ( p.titleKeywords.count() != 0 ) { + TQString op = (p.requireAllTitleWords) ? "AND " : "OR "; + + TQString condition = "("; + for ( TQStringList::const_iterator it = p.titleKeywords.begin(); it != p.titleKeywords.end();) { + condition += "r.title LIKE '%"+escapeAndEncode(*it)+"%' "; + if ( ++it != p.titleKeywords.end() ) { + condition += op; + } + } + condition += ")"; + conditionList << condition; + } + + if ( p.instructionsKeywords.count() != 0 ) { + TQString op = (p.requireAllInstructionsWords) ? "AND " : "OR "; + + TQString condition = "("; + for ( TQStringList::const_iterator it = p.instructionsKeywords.begin(); it != p.instructionsKeywords.end();) { + condition += "r.instructions LIKE '%"+escapeAndEncode(*it)+"%' "; + if ( ++it != p.instructionsKeywords.end() ) { + condition += op; + } + } + condition += ")"; + conditionList << condition; + } + + if ( !p.prep_time.isNull() ) { + TQString op; + switch ( p.prep_param ) { + case 0: op = "<= "+p.prep_time.toString( "'hh:mm:ss'" ); break; + case 1: //TODO: have a configurable 'about'. It tests within 15 minutes for now. + TQTime lower = p.prep_time; lower.addSecs( 60*15 ); + TQTime upper = p.prep_time; upper.addSecs( 60*-15 ); + op = "BETWEEN "+lower.toString( "'hh:mm:ss'" )+" AND "+upper.toString( "'hh:mm:ss'" ); + break; + } + conditionList << "r.prep_time "+op; + } + + if ( p.servings > 0 ) { + TQString op; + switch ( p.servings_param ) { + case 0: op = "> "+TQString::number(p.servings); break; + case 1: op = "< "+TQString::number(p.servings); break; + case 2: op = "BETWEEN "+TQString::number(p.servings-5)+" AND "+TQString::number(p.servings+5); break; + } + conditionList << "r.yield_amount "+op; + } + + if ( p.createdDateBegin.isValid() ) { + if ( p.createdDateEnd.isValid() ) { + conditionList << "r.ctime >= '"+p.createdDateBegin.toString(TQt::ISODate)+"'"; + conditionList << "r.ctime <= '"+p.createdDateEnd.toString(TQt::ISODate)+"'"; + } + else { + if ( p.createdDateBegin.time().isNull() ) { //we just want something on a particular date, not time + TQDateTime end = p.createdDateBegin.addDays(1); + conditionList << "r.ctime >= '"+p.createdDateBegin.toString(TQt::ISODate)+"'"; + conditionList << "r.ctime <= '"+end.toString(TQt::ISODate)+"'"; + } + else //use the exact time + conditionList << "r.ctime = '"+p.createdDateBegin.toString(TQt::ISODate)+"'"; + } + } + + if ( p.modifiedDateBegin.isValid() ) { + if ( p.modifiedDateEnd.isValid() ) { + conditionList << "r.mtime >= '"+p.modifiedDateBegin.toString(TQt::ISODate)+"'"; + conditionList << "r.mtime <= '"+p.modifiedDateEnd.toString(TQt::ISODate)+"'"; + } + else { + if ( p.modifiedDateBegin.time().isNull() ) { //we just want something on a particular date, not time + TQDateTime end = p.modifiedDateBegin.addDays(1); + conditionList << "r.mtime >= '"+p.modifiedDateBegin.toString(TQt::ISODate)+"'"; + conditionList << "r.mtime <= '"+end.toString(TQt::ISODate)+"'"; + } + else //use the exact time + conditionList << "r.mtime = '"+p.modifiedDateBegin.toString(TQt::ISODate)+"'"; + } + } + + if ( p.accessedDateBegin.isValid() ) { + if ( p.accessedDateEnd.isValid() ) { + conditionList << "r.atime >= '"+p.accessedDateBegin.toString(TQt::ISODate)+"'"; + conditionList << "r.atime <= '"+p.accessedDateEnd.toString(TQt::ISODate)+"'"; + } + else { + if ( p.accessedDateBegin.time().isNull() ) { //we just want something on a particular date, not time + TQDateTime end = p.accessedDateBegin.addDays(1); + conditionList << "r.atime >= '"+p.accessedDateBegin.toString(TQt::ISODate)+"'"; + conditionList << "r.atime <= '"+end.toString(TQt::ISODate)+"'"; + } + else //use the exact time + conditionList << "r.atime = '"+p.accessedDateBegin.toString(TQt::ISODate)+"'"; + } + } + + TQString wholeQuery = "SELECT r.id FROM recipes r" + +TQString(tableList.count()!=0?","+tableList.join(","):"") + +TQString(conditionList.count()!=0?" WHERE "+conditionList.join(" AND "):""); + + kdDebug()<<"calling: "<<wholeQuery<<endl; + return wholeQuery+";"; +} + +//These are helper functions solely for use by the USDA data importer +void getIngredientNameAndID( std::multimap<int, TQString> * ); +int createUnit( const TQString &name, Unit::Type, RecipeDB* ); +int createIngredient( const TQString &name, int unit_g_id, int unit_mg_id, RecipeDB*, bool do_checks ); +void create_properties( RecipeDB* ); + +void RecipeDB::importUSDADatabase() +{ + //check if the data file even exists before we do anything + TQString abbrev_file = locate( "appdata", "data/abbrev.txt" ); + if ( abbrev_file.isEmpty() ) { + kdDebug() << "Unable to find abbrev.txt data file." << endl; + return ; + } + + TQFile file( abbrev_file ); + if ( !file.open( IO_ReadOnly ) ) { + kdDebug() << "Unable to open data file: " << abbrev_file << endl; + return ; + } + + create_properties( this ); + + std::multimap<int, TQString> *ings_and_ids = new std::multimap<int, TQString>; + getIngredientNameAndID( ings_and_ids ); + + TQTextStream stream( &file ); + TQValueList<ingredient_nutrient_data> *data = new TQValueList<ingredient_nutrient_data>; + + kdDebug() << "Parsing abbrev.txt" << endl; + while ( !stream.atEnd() ) { + TQStringList fields = TQStringList::split( "^", stream.readLine(), true ); + + int id = fields[ 0 ].mid( 1, fields[ 0 ].length() - 2 ).toInt(); + + std::multimap<int, TQString>::iterator current_pair; + while ( ( current_pair = ings_and_ids->find( id ) ) != ings_and_ids->end() ) //there may be more than one ingredients with the same id + { + ingredient_nutrient_data current_ing; + current_ing.name = ( *current_pair ).second.latin1(); + + for ( int i = 2; i < TOTAL_USDA_PROPERTIES + 2; i++ ) //properties start at the third field (index 2) + current_ing.data << fields[ i ].toDouble(); + + Weight w; + w.weight = fields[ TOTAL_USDA_PROPERTIES + 2 ].toDouble(); + + TQString amountAndWeight = fields[ TOTAL_USDA_PROPERTIES + 3 ].mid( 1, fields[ TOTAL_USDA_PROPERTIES + 3 ].length() - 2 ); + if ( !amountAndWeight.isEmpty() ) { + int spaceIndex = amountAndWeight.find(" "); + w.perAmount = amountAndWeight.left(spaceIndex).toDouble(); + + TQString perAmountUnit = amountAndWeight.right(amountAndWeight.length()-spaceIndex-1); + + if ( parseUSDAUnitAndPrep( perAmountUnit, w.perAmountUnit, w.prepMethod ) ) + current_ing.weights << w; + } + + w = Weight(); + w.weight = fields[ TOTAL_USDA_PROPERTIES + 4 ].toDouble(); + amountAndWeight = fields[ TOTAL_USDA_PROPERTIES + 5 ].mid( 1, fields[ TOTAL_USDA_PROPERTIES + 5 ].length() - 2 ); + if ( !amountAndWeight.isEmpty() ) { + int spaceIndex = amountAndWeight.find(" "); + w.perAmount = amountAndWeight.left(spaceIndex).toDouble(); + TQString perAmountUnit = amountAndWeight.right(amountAndWeight.length()-spaceIndex-1); + + if ( parseUSDAUnitAndPrep( perAmountUnit, w.perAmountUnit, w.prepMethod ) ) + current_ing.weights << w; + } + + current_ing.usda_id = id; + + data->append( current_ing ); + + ings_and_ids->erase( current_pair ); + } + } + + delete ings_and_ids; + + //there's 13009 lines in the weight file + emit progressBegin( data->count(), i18n( "Nutrient Import" ), i18n( "Importing USDA nutrient data" ) ); + + //if there is no data in the database, we can really speed this up with this + bool do_checks = true; + { + ElementList ing_list; + loadIngredients( &ing_list ); + + if ( ing_list.count() == 0 ) { + kdDebug()<<"Found an empty database... enabling fast nutrient import"<<endl; + do_checks = false; + } + } + + //since there are only two units used, lets just create them and store their id for speed + int unit_g_id = createUnit( "g", Unit::Mass, this ); + int unit_mg_id = createUnit( "mg", Unit::Mass, this ); + + TQValueList<ingredient_nutrient_data>::const_iterator it; + TQValueList<ingredient_nutrient_data>::const_iterator data_end = data->end(); + const int total = data->count(); + int counter = 0; + + for ( it = data->begin(); it != data_end; ++it ) { + counter++; + kdDebug() << "Inserting (" << counter << " of " << total << "): " << ( *it ).name << endl; + + if ( haltOperation ) { haltOperation=false; break;} + emit progress(); + + int assigned_id = createIngredient( ( *it ).name, unit_g_id, unit_mg_id, this, do_checks ); + + //for now, only check if there is any info on the ingredient to see whether or not we will import this data, + //because checking to see that each property exists is quite slow + IngredientPropertyList ing_properties; + if ( do_checks ) loadProperties( &ing_properties, assigned_id ); + if ( ing_properties.count() == 0 ) //ingredient doesn't already have any properties + { + TQValueList<double>::const_iterator property_it; + TQValueList<double>::const_iterator property_end = ( *it ).data.end(); + int i = 0; + for ( property_it = ( *it ).data.begin(); property_it != property_end; ++property_it, ++i ) + addPropertyToIngredient( assigned_id, property_data_list[ i ].id, ( *property_it ) / 100.0, unit_g_id ); + } + + WeightList existingWeights = ingredientWeightUnits( assigned_id ); + const WeightList weights = (*it).weights; + for ( WeightList::const_iterator weight_it = weights.begin(); weight_it != weights.end(); ++weight_it ) { + Weight w = *weight_it; + w.perAmountUnitID = createUnit( w.perAmountUnit, Unit::Other, this ); + w.weightUnitID = unit_g_id; + w.ingredientID = assigned_id; + + //TODO optimze by creating all prep methods and storing them for faster non-db access + if ( !w.prepMethod.isEmpty() ) { + int prepID = findExistingPrepByName( w.prepMethod ); + if ( prepID == -1 ) { + createNewPrepMethod( w.prepMethod ); + prepID = lastInsertID(); + } + w.prepMethodID = prepID; + } + + bool exists = false; + for ( WeightList::const_iterator it = existingWeights.begin(); it != existingWeights.end(); ++it ) { + if ( (*it).perAmountUnitID == w.perAmountUnitID && (*it).prepMethodID == w.prepMethodID ) { + exists = true; + break; + } + } + if ( exists ) + continue; + + addIngredientWeight( w ); + } + } + + delete data; + + kdDebug() << "USDA data import successful" << endl; + + emit progressDone(); +} + +void getIngredientNameAndID( std::multimap<int, TQString> *data ) +{ + for ( int i = 0; !ingredient_data_list[ i ].name.isEmpty(); i++ ) + data->insert( std::make_pair( ingredient_data_list[ i ].usda_id, ingredient_data_list[ i ].name ) ); +} + +int createIngredient( const TQString &name, int unit_g_id, int unit_mg_id, RecipeDB *database, bool do_checks ) +{ + bool ingredientExisted = true; + int assigned_id = -1; + if ( do_checks ) + assigned_id = database->findExistingIngredientByName( name ); + + if ( assigned_id == -1 ) { + ingredientExisted = false; + database->createNewIngredient( name ); + assigned_id = database->lastInsertID(); + } + + if ( !ingredientExisted || !database->ingredientContainsUnit( assigned_id, unit_g_id ) ) + database->addUnitToIngredient( assigned_id, unit_g_id ); + + if ( !ingredientExisted || !database->ingredientContainsUnit( assigned_id, unit_mg_id ) ) + database->addUnitToIngredient( assigned_id, unit_mg_id ); + + return assigned_id; +} + +int createUnit( const TQString &name, Unit::Type type, RecipeDB *database ) +{ + int assigned_id = database->findExistingUnitByName( name ); + + if ( assigned_id == -1 ) //create unit since it doesn't exist + { + Unit unit(name, name); + unit.type = type; + database->createNewUnit( unit ); + assigned_id = database->lastInsertID(); + } + //keep what the user specified if the type here is Other + else if ( type != Unit::Other ) { + Unit unit = database->unitName(assigned_id); + if ( unit.type != type ) { + unit.type = type; + database->modUnit( unit ); + } + } + + return assigned_id; +} + +void create_properties( RecipeDB *database ) +{ + IngredientPropertyList property_list; + database->loadProperties( &property_list ); + + for ( int i = 0; !property_data_list[ i ].name.isEmpty(); i++ ) { + property_data_list[ i ].id = property_list.findByName( property_data_list[ i ].name ); + if ( property_data_list[ i ].id == -1 ) //doesn't exist, so insert it and set property_data_list[i].id + { + database->addProperty( property_data_list[ i ].name, property_data_list[ i ].unit ); + property_data_list[ i ].id = database->lastInsertID(); + } + } +} + +bool parseUSDAUnitAndPrep( const TQString &string, TQString &unit, TQString &prep ) +{ + int commaIndex = string.find(","); + TQString unitPart = string.left(commaIndex); + TQString prepPart = string.right(string.length()-commaIndex-2).stripWhiteSpace(); + + bool acceptable = false; + for ( int i = 0; unit_data_list[ i ].name; ++i ) { + if ( unitPart == unit_data_list[ i ].name || unitPart == unit_data_list[ i ].plural ) + acceptable = true; + } + if ( !acceptable ) + return false; + + acceptable = false; + if ( prepPart.isEmpty() ) + acceptable = true; + else { + for ( int i = 0; prep_data_list[ i ]; ++i ) { + if ( prepPart == prep_data_list[ i ] ) + acceptable = true; + } + } + if ( !acceptable ) + prepPart = TQString::null; + + unit = unitPart; + prep = prepPart; + return true; +} + +#include "recipedb.moc" diff --git a/src/backends/recipedb.h b/src/backends/recipedb.h new file mode 100644 index 0000000..cf43127 --- /dev/null +++ b/src/backends/recipedb.h @@ -0,0 +1,397 @@ +/*************************************************************************** +* Copyright (C) 2003 * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* * +* Copyright (C) 2003-2006 Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef RECIPEDB_H +#define RECIPEDB_H + +#include <tqobject.h> +#include <tqstring.h> +#include <tqvaluelist.h> + +#include <dcopclient.h> + +#include "krecipesdbiface.h" + +#include "datablocks/recipe.h" +#include "datablocks/recipelist.h" +#include "datablocks/elementlist.h" +#include "datablocks/ingredientpropertylist.h" +#include "datablocks/unitratiolist.h" +#include "datablocks/unit.h" + +#define DEFAULT_DB_NAME "Krecipes" + +/** +@author Unai Garro +*/ + +class TDEProcess; +class TQTextStream; + +class CategoryTree; +class RecipeSearchParameters; +class Weight; +class WeightList; + +typedef struct +{ + TQValueList <int> recipeIdList; + IngredientList ilist; +} +RecipeIngredientList; + +class RecipeDB: public TQObject, virtual public KrecipesDBIface +{ + TQ_OBJECT + +public: + RecipeDB(); + virtual ~RecipeDB(); + + virtual void connect( bool create_db = true, bool create_tables = true ) = 0; + + void importSamples(); + + bool backup( const TQString &file ){ return backup(file,0); } + bool backup( const TQString &file, TQString *errMsg = 0 ); + bool restore( const TQString &file, TQString *errMsg = 0 ); + + // Error handling (passive) + bool dbOK; + TQString dbErr; + + enum RecipeItems { + None = 0, + NamesOnly = 256, + Noatime = 1024, + Photo = 1, + Instructions = 2, + Ingredients = 4, + Authors = 8, + Categories = 16, + PrepTime = 32, + Yield = 64, + Title = 128, + Meta = 512, + Ratings = 2048, + Properties = 4096, + IngredientAmounts = 8192, + All = 0xFFFF ^ NamesOnly ^ Noatime + }; + + enum ConversionStatus { + Success, + MissingUnitConversion, + MissingIngredientWeight, + MissingIngredient, + InvalidTypes, + MismatchedPrepMethod, + MismatchedPrepMethodUsingApprox + }; + +public slots: + void cancelOperation(){ haltOperation = true; } + +signals: + void progressBegin(int,const TQString &c=TQString::null,const TQString &t=TQString::null,int rate=1); + void progressDone(); + void progress(); + + void authorCreated( const Element & ); + void authorRemoved( int id ); + + void categoryCreated( const Element &, int parent_id ); + void categoryRemoved( int id ); + void categoryModified( const Element & ); + void categoryModified( int id, int parent_id ); + void categoriesMerged( int id1, int id2 ); + + void ingGroupCreated( const Element & ); + void ingGroupRemoved( int id ); + + void ingredientCreated( const Element & ); + void ingredientRemoved( int id ); + + void prepMethodCreated( const Element & ); + void prepMethodRemoved( int id ); + + void propertyCreated( const IngredientProperty & ); + void propertyRemoved( int id ); + + void unitCreated( const Unit & ); + void unitRemoved( int id ); + + void ratingCriteriaCreated( const Element & ); + + void recipeCreated( const Element &, const ElementList &categories ); + void recipeRemoved( int id ); + void recipeRemoved( int id, int cat_id ); + void recipeModified( const Element &, const ElementList &categories ); + + // Public methods +public: + /** Returns a database object of the given type or NULL upon failure. + * This function should be called to create a new database, rather + * than directly calling the constructor of a specific backend. + */ + static RecipeDB* createDatabase( const TQString &dbType, + const TQString &host, + const TQString &user, + const TQString &pass, + const TQString &DBname, + int port, + const TQString &file = TQString::null ); + + /** Convenience method. Calls the above with arguments from TDEConfig. */ + static RecipeDB* createDatabase( const TQString &dbType, const TQString &file = TQString::null ); + + virtual void addIngredientWeight( const Weight & ) = 0; + virtual void addProperty( const TQString &name, const TQString &units ) = 0; + virtual void addPropertyToIngredient( int ingredientID, int propertyID, double amount, int perUnitsID ) = 0; + virtual void addUnitToIngredient( int ingredientID, int unitID ) = 0; + + virtual void categorizeRecipe( int recipeID, const ElementList &categoryList ) = 0; + virtual void changePropertyAmountToIngredient( int ingredientID, int propertyID, double amount, int per_units ) = 0; + + virtual void createNewAuthor( const TQString &authorName ) = 0; + virtual void createNewCategory( const TQString &categoryName, int parent_id = -1 ) = 0; + virtual void createNewIngGroup( const TQString &name ) = 0; + virtual void createNewIngredient( const TQString &ingredientName ) = 0; + virtual void createNewPrepMethod( const TQString &prepMethodName ) = 0; + virtual void createNewRating( const TQString &name ) = 0; + virtual void createNewUnit( const Unit &unit ) = 0; + virtual void createNewYieldType( const TQString &type ) = 0; + + virtual void emptyData( void ) = 0; + virtual void empty( void ) = 0; + + virtual int findExistingAuthorByName( const TQString& name ) = 0; + virtual int findExistingCategoryByName( const TQString& name ) = 0; + virtual int findExistingIngredientGroupByName( const TQString& name ) = 0; + virtual int findExistingIngredientByName( const TQString& name ) = 0; + virtual int findExistingPrepByName( const TQString& name ) = 0; + virtual int findExistingPropertyByName( const TQString& name ) = 0; + virtual int findExistingRatingByName( const TQString& name ) = 0; + virtual int findExistingRecipeByName( const TQString& name ) = 0; + virtual int findExistingUnitByName( const TQString& name ) = 0; + virtual int findExistingYieldTypeByName( const TQString& name ) = 0; + virtual void findIngredientUnitDependancies( int ingredientID, int unitID, ElementList *recipes, ElementList *ingredientInfo ) = 0; + virtual void findIngredientDependancies( int ingredientID, ElementList *recipes ) = 0; + virtual void findPrepMethodDependancies( int prepMethodID, ElementList *recipes ) = 0; + virtual void findUnitDependancies( int unitID, ElementList *properties, ElementList *recipes, ElementList *weights ) = 0; + virtual void findUseOfIngGroupInRecipes( ElementList *results, int groupID ) = 0; + virtual void findUseOfCategoryInRecipes( ElementList *results, int catID ) = 0; + virtual void findUseOfAuthorInRecipes( ElementList *results, int authorID ) = 0; + + void getIDList( const CategoryTree *categoryTree, TQStringList &ids ); + virtual TQString getUniqueRecipeTitle( const TQString &recipe_title ) = 0; + virtual void givePermissions( const TQString &dbName, const TQString &username, const TQString &password = TQString::null, const TQString &clientHost = "localhost" ) = 0; + + void importUSDADatabase(); + + virtual bool ingredientContainsProperty( int ingredientID, int propertyID, int perUnitsID ) = 0; + virtual bool ingredientContainsUnit( int ingredientID, int unitID ) = 0; + + void initializeData( void ); + + virtual int lastInsertID() = 0; + + virtual void loadAuthors( ElementList *list, int limit = -1, int offset = 0 ) = 0; + virtual void loadCategories( CategoryTree *list, int limit = -1, int offset = 0, int parent_id = -1, bool recurse = true ) = 0; + void loadCachedCategories( CategoryTree **list, int limit, int offset, int parent_id, bool recurse ); + virtual void loadCategories( ElementList *list, int limit = -1, int offset = 0 ) = 0; + virtual void loadIngredientGroups( ElementList *list ) = 0; + virtual void loadIngredients( ElementList *list, int limit = -1, int offset = 0 ) = 0; + virtual void loadPossibleUnits( int ingredientID, UnitList *list ) = 0; + virtual void loadPrepMethods( ElementList *list, int limit = -1, int offset = 0 ) = 0; + virtual void loadProperties( IngredientPropertyList *list, int ingredientID = -2 ) = 0; // Loads the list of possible properties by default, all the ingredient properties with -1, and the ingredients of given property if id>=0 + void loadRecipe( Recipe *recipe, int items, int id ); + + virtual void loadRatingCriterion( ElementList *list, int limit = -1, int offset = 0 ) = 0; + /** Load all recipes with the ids in @param ids into the @ref RecipeList @param recipes */ + virtual void loadRecipes( RecipeList *, int items = All, TQValueList<int> ids = TQValueList<int>()/*, KProgressDialog *progress_dlg = 0*/ ) = 0; + virtual void loadRecipeList( ElementList *list, int categoryID = -1, bool recursive = false ) = 0; + virtual void loadUncategorizedRecipes( ElementList *list ) = 0; + virtual void loadUnits( UnitList *list, Unit::Type = Unit::All, int limit = -1, int offset = 0 ) = 0; + virtual void loadUnitRatios( UnitRatioList *ratioList, Unit::Type ) = 0; + virtual void loadYieldTypes( ElementList *list, int limit = -1, int offset = 0 ) = 0; + + /** Change all instances of authors with id @param id2 to @param id1 */ + virtual void mergeAuthors( int id1, int id2 ) = 0; + + /** Change all instances of categories with id @param id2 to @param id1 */ + virtual void mergeCategories( int id1, int id2 ) = 0; + + virtual void mergeIngredientGroups( int id1, int id2 ) = 0; + + /** Change all instances of ingredients with id @param id2 to @param id1 */ + virtual void mergeIngredients( int id1, int id2 ) = 0; + + /** Change all instances of units with id @param id2 to @param id1 */ + virtual void mergeUnits( int id1, int id2 ) = 0; + + /** Change all instances of prep methods with id @param id2 to @param id1 */ + virtual void mergePrepMethods( int id1, int id2 ) = 0; + + virtual void mergeProperties( int id1, int id2 ) = 0; + + + virtual void modIngredientGroup( int ingredientID, const TQString &newLabel ) = 0; + /** + * set newLabel for ingredientID + */ + virtual void modIngredient( int ingredientID, const TQString &newLabel ) = 0; + /** + * set newLabel for unitID + */ + virtual void modUnit( const Unit &unit ) = 0; + /** + * set newLabel for categoryID + */ + virtual void modCategory( int categoryID, const TQString &newLabel ) = 0; + virtual void modCategory( int categoryID, int new_parent_id ) = 0; + /** + * set newLabel for authorID + */ + virtual void modAuthor( int authorID, const TQString &newLabel ) = 0; + + virtual void modPrepMethod( int prepMethodID, const TQString &newLabel ) = 0; + + virtual void modProperty( int propertyID, const TQString &newLabel ) = 0; + + virtual TQString recipeTitle( int recipeID ) = 0; + + virtual void removeAuthor( int categoryID ) = 0; + virtual void removeCategory( int categoryID ) = 0; + virtual void removeIngredientGroup( int ingredientID ) = 0; + virtual void removeIngredient( int ingredientID ) = 0; + virtual void removeIngredientWeight( int id ) = 0; + virtual void removePrepMethod( int prepMethodID ) = 0; + virtual void removeProperty( int propertyID ) = 0; + virtual void removePropertyFromIngredient( int ingredientID, int propertyID, int perUnitID ) = 0; + virtual void removeRecipe( int id ) = 0; + virtual void removeRecipeFromCategory( int ingredientID, int categoryID ) = 0; + virtual void removeUnit( int unitID ) = 0; + virtual void removeUnitFromIngredient( int ingredientID, int unitID ) = 0; + virtual void removeUnitRatio( int unitID1, int unitID2 ) = 0; + + virtual void saveRecipe( Recipe *recipe ) = 0; + virtual void saveUnitRatio( const UnitRatio *ratio ) = 0; + virtual void search( RecipeList *list, int items, const RecipeSearchParameters ¶meters ) = 0; + + /** @returns true on success, false otherwise */ + ConversionStatus convertIngredientUnits( const Ingredient &from, const Unit &to, Ingredient &result ); + virtual double unitRatio( int unitID1, int unitID2 ) = 0; + + /** @returns the number of grams in the given amount of the ingredient, or -1 on failure */ + virtual double ingredientWeight( const Ingredient &ing, bool *wasApproximated = 0 ) = 0; + virtual WeightList ingredientWeightUnits( int ingID ) = 0; + + virtual TQString escapeAndEncode( const TQString &s ) const = 0; + virtual TQString unescapeAndDecode( const TQCString &s ) const = 0; + + virtual TQString categoryName( int ID ) = 0; + virtual TQString ingredientName( int ID ) = 0; + virtual TQString prepMethodName( int ID ) = 0; + virtual IngredientProperty propertyName( int ID ) = 0; + virtual Unit unitName( int ID ) = 0; + + virtual int categoryTopLevelCount() = 0; + virtual int getCount( const TQString &table_name ) = 0; + int authorCount(); + int ingredientCount(); + int prepMethodCount(); + int unitCount(); + int categoryCount(); + + virtual bool checkIntegrity( void ) = 0; + + virtual void createTable( const TQString &tableName ) = 0; + virtual void splitCommands( TQString& s, TQStringList& sl ) = 0; + + virtual float databaseVersion( void ) = 0; + + int maxAuthorNameLength() const + { + return 50; + } + int maxCategoryNameLength() const + { + return 40; + } + int maxIngredientNameLength() const + { + return 50; + } + int maxIngGroupNameLength() const + { + return 50; + } + int maxRecipeTitleLength() const + { + return 200; + } + int maxUnitNameLength() const + { + return 20; + } + int maxPrepMethodNameLength() const + { + return 20; + } + int maxPropertyNameLength() const + { + return 20; + } + int maxYieldTypeLength() const + { + return 20; + } + + virtual bool ok() + { + return ( dbOK ); + } + virtual TQString err() + { + return ( dbErr ); + } + + void updateCategoryCache( int limit ); + void clearCategoryCache(); + +protected: + virtual void portOldDatabases( float version ) = 0; + virtual TQStringList backupCommand() const = 0; + virtual TQStringList restoreCommand() const = 0; + + //Use these with caution: SQL for one backend might not work on another! + void execSQL( TQTextStream &stream ); + virtual void execSQL( const TQString & ) = 0; + + TQString buildSearchQuery( const RecipeSearchParameters ¶meters ) const; + + double latestDBVersion() const; + TQString krecipes_version() const; + + CategoryTree *m_categoryCache; + +private: + TQTextStream *dumpStream; + bool haltOperation; + +private slots: + void processDumpOutput( TDEProcess *, char *buffer, int buflen ); +}; + +#endif diff --git a/src/backends/searchparameters.h b/src/backends/searchparameters.h new file mode 100644 index 0000000..643cb29 --- /dev/null +++ b/src/backends/searchparameters.h @@ -0,0 +1,63 @@ +/*************************************************************************** +* Copyright (C) 2005 * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef SEARCHPARAMETERS_H +#define SEARCHPARAMETERS_H + +#include <tqstring.h> +#include <tqstringlist.h> +#include <tqdatetime.h> + +class RecipeSearchParameters +{ +public: + RecipeSearchParameters() : servings(-1)/*, averageRating(-1), averageRatingOffset(0)*/ + {} + + TQStringList titleKeywords; + bool requireAllTitleWords; + + TQStringList instructionsKeywords; + bool requireAllInstructionsWords; + + TQStringList ingsOr; + TQStringList catsOr; + TQStringList authorsOr; + + TQTime prep_time; + + /** 0 -> greater than given time + * 1 -> less than given time + * 2 -> about given time + */ + int prep_param; + + int servings; + + /** 0 -> greater than given time + * 1 -> less than given time + * 2 -> about given time + */ + int servings_param; + + TQDateTime createdDateBegin; + TQDateTime createdDateEnd; + TQDateTime modifiedDateBegin; + TQDateTime modifiedDateEnd; + TQDateTime accessedDateBegin; + TQDateTime accessedDateEnd; + + //RatingCriteriaList criteriaList; + //double averageRating; + //double averageRatingOffset; +}; + + +#endif diff --git a/src/backends/usda_ingredient_data.h b/src/backends/usda_ingredient_data.h new file mode 100644 index 0000000..9fb4b8a --- /dev/null +++ b/src/backends/usda_ingredient_data.h @@ -0,0 +1,486 @@ +/*************************************************************************** +* Copyright (C) 2003 * +* * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef USDA_INGREDIENT_DATA_H +#define USDA_INGREDIENT_DATA_H + +#include <tdelocale.h> + +struct ingredient_data +{ + TQString name; + int usda_id; +}; + +static ingredient_data ingredient_data_list[] = { + {I18N_NOOP( "active baker's yeast" ), 18375}, + {I18N_NOOP( "all-purpose flour" ), 20081}, + {I18N_NOOP( "allspice" ), 2001}, + {I18N_NOOP( "almond extract" ), 12071}, + {I18N_NOOP( "almonds" ), 12061}, + {I18N_NOOP( "apple juice" ), 9016}, + {I18N_NOOP( "apple slices" ), 9008}, + {I18N_NOOP( "apples" ), 9003}, + {I18N_NOOP( "applesauce" ), 9402}, + {I18N_NOOP( "applesauce, unsweetened" ), 9019}, + {I18N_NOOP( "apricot jam" ), 19719}, + {I18N_NOOP( "bacon" ), 10123}, + {I18N_NOOP( "baking potato" ), 11362}, + {I18N_NOOP( "baking powder" ), 18369}, + {I18N_NOOP( "baking soda" ), 18372}, + {I18N_NOOP( "bananas" ), 9040}, + {I18N_NOOP( "barley" ), 20004}, + {I18N_NOOP( "basil" ), 2003}, + {I18N_NOOP( "bay leaf" ), 2004}, + {I18N_NOOP( "bean sprouts" ), 11248}, + {I18N_NOOP( "beans, black" ), 16014}, + {I18N_NOOP( "beef bottom round" ), 13399}, + {I18N_NOOP( "beef bouillon" ), 6075}, + {I18N_NOOP( "beef broth" ), 6008}, + {I18N_NOOP( "beef stock" ), 6476}, + {I18N_NOOP( "beef, brisket" ), 13022}, + {I18N_NOOP( "beer" ), 14003}, + {I18N_NOOP( "bell peppers (red, green, yellow)" ), 11333}, + {I18N_NOOP( "black pepper" ), 2030}, + {I18N_NOOP( "boiling water" ), 14429}, + {I18N_NOOP( "bologna" ), 7008}, + {I18N_NOOP( "bourbon" ), 14551}, + {I18N_NOOP( "bran flakes" ), 8153}, + {I18N_NOOP( "bread" ), 18069}, + {I18N_NOOP( "bread crumbs" ), 18079}, + {I18N_NOOP( "bread cubes" ), 18079}, + {I18N_NOOP( "broccoli" ), 11090}, + {I18N_NOOP( "broccoli (frozen-thawed)" ), 11092}, + {I18N_NOOP( "broccoli spears, frozen" ), 11094}, + {I18N_NOOP( "broccoli, frozen" ), 11092}, + {I18N_NOOP( "brown rice" ), 20040}, + {I18N_NOOP( "brown sugar" ), 19334}, + {I18N_NOOP( "bulgur" ), 20012}, + {I18N_NOOP( "butter" ), 1145}, + {I18N_NOOP( "butter (1/2 stick)" ), 1145}, + {I18N_NOOP( "butter or margarine" ), 1145}, + {I18N_NOOP( "buttermilk" ), 1088}, + {I18N_NOOP( "cabbage" ), 11109}, + {I18N_NOOP( "canned apple slices" ), 19312}, + {I18N_NOOP( "canned beans" ), 16006}, + {I18N_NOOP( "canned beef" ), 13934}, + {I18N_NOOP( "canned black beans" ), 16018}, + {I18N_NOOP( "canned carrot slices" ), 11128}, + {I18N_NOOP( "canned cooked squid" ), 15175}, + {I18N_NOOP( "canned corn" ), 11174}, + {I18N_NOOP( "canned crushed tomatoes" ), 11693}, + {I18N_NOOP( "canned green beans" ), 11056}, + {I18N_NOOP( "canned green peas" ), 11308}, + {I18N_NOOP( "canned jalapeno pepper" ), 11632}, + {I18N_NOOP( "canned kidney beans" ), 16029}, + {I18N_NOOP( "canned pinto beans" ), 16044}, + {I18N_NOOP( "canned potato" ), 11376}, + {I18N_NOOP( "canned red pepper" ), 2031}, + {I18N_NOOP( "canned red tart cherries in water" ), 9065}, + {I18N_NOOP( "canned sweet potatoes" ), 11514}, + {I18N_NOOP( "canned tomatoes" ), 11693}, + {I18N_NOOP( "canned wax beans" ), 16029}, + {I18N_NOOP( "canned white beans" ), 16051}, + {I18N_NOOP( "canned whole kernel corn" ), 11771}, + {I18N_NOOP( "carrot" ), 11124}, + {I18N_NOOP( "carrot slices, canned" ), 11128}, + {I18N_NOOP( "carrots" ), 11124}, + {I18N_NOOP( "carrots, frozen" ), 11130}, + {I18N_NOOP( "carrots, peeled and chopped" ), 11124}, + {I18N_NOOP( "catsup" ), 11935}, + {I18N_NOOP( "cauliflower, frozen" ), 11137}, + {I18N_NOOP( "cayenne" ), 2031}, + {I18N_NOOP( "celery" ), 11143}, + {I18N_NOOP( "celery leaf" ), 11143}, + {I18N_NOOP( "celery salt" ), 2007}, + {I18N_NOOP( "celery seed" ), 2007}, + {I18N_NOOP( "cheddar cheese" ), 1009}, + {I18N_NOOP( "cheese" ), 1009}, + {I18N_NOOP( "cherry pie filling" ), 19314}, + {I18N_NOOP( "chicken" ), 5011}, + {I18N_NOOP( "chicken (1-2 lb)" ), 5011}, + {I18N_NOOP( "chicken bouillon" ), 6081}, + {I18N_NOOP( "chicken breast" ), 5062}, + {I18N_NOOP( "chicken broth" ), 6013}, + {I18N_NOOP( "chicken pieces" ), 5008}, + {I18N_NOOP( "chicken stock" ), 6413}, + {I18N_NOOP( "chicken thigh" ), 5080}, + {I18N_NOOP( "chicken, cooked" ), 5011}, + {I18N_NOOP( "chickens, whole" ), 5123}, + {I18N_NOOP( "chili" ), 16059}, + {I18N_NOOP( "chili powder" ), 2009}, + {I18N_NOOP( "chinese turnip" ), 11564}, + {I18N_NOOP( "chives" ), 11615}, + {I18N_NOOP( "chocolate chip" ), 19139}, + {I18N_NOOP( "chocolate pudding mix (instant)" ), 19123}, + {I18N_NOOP( "chopped capers" ), 2054}, + {I18N_NOOP( "chopped carrot" ), 11124}, + {I18N_NOOP( "chopped celery" ), 11143}, + {I18N_NOOP( "chopped chives" ), 11156}, + {I18N_NOOP( "chopped green chilies" ), 11980}, + {I18N_NOOP( "chopped onion" ), 11282}, + {I18N_NOOP( "chopped onions" ), 11282}, + {I18N_NOOP( "chopped onions, frozen" ), 11287}, + {I18N_NOOP( "chopped parsley" ), 11297}, + {I18N_NOOP( "chopped pimientos" ), 11943}, + {I18N_NOOP( "chopped walnuts" ), 12154}, + {I18N_NOOP( "cider vinegar" ), 2048}, + {I18N_NOOP( "cilantro" ), 11971}, + {I18N_NOOP( "cinnamon" ), 2010}, + {I18N_NOOP( "clear honey" ), 19296}, + {I18N_NOOP( "cloves" ), 2011}, + {I18N_NOOP( "cloves garlic" ), 11215}, + {I18N_NOOP( "cloves or allspice" ), 2011}, + {I18N_NOOP( "cocoa" ), 19165}, + {I18N_NOOP( "coconut" ), 12104}, + {I18N_NOOP( "coconut milk (canned)" ), 12118}, + {I18N_NOOP( "cold butter" ), 1145}, + {I18N_NOOP( "cold water" ), 14429}, + {I18N_NOOP( "condensed milk (sweetened)" ), 1095}, + {I18N_NOOP( "cooked chicken" ), 5124}, + {I18N_NOOP( "cooked chicken breast" ), 5065}, + {I18N_NOOP( "cooked ham" ), 10151}, + {I18N_NOOP( "cooked turkey" ), 5182}, + {I18N_NOOP( "cooked wild rice" ), 20089}, + {I18N_NOOP( "cookie crust (chocolate - 8-9 in)" ), 18398}, + {I18N_NOOP( "Cool Whip" ), 1054}, + {I18N_NOOP( "corn chips" ), 19003}, + {I18N_NOOP( "corn meal" ), 20020}, + {I18N_NOOP( "corn oil" ), 4518}, + {I18N_NOOP( "corn starch" ), 20027}, + {I18N_NOOP( "corn syrup" ), 19350}, + {I18N_NOOP( "cornmeal" ), 20022}, + {I18N_NOOP( "cornstarch" ), 20027}, + {I18N_NOOP( "cottage cheese" ), 1015}, + {I18N_NOOP( "cream (heavy)" ), 1053}, + {I18N_NOOP( "cream cheese" ), 1017}, + {I18N_NOOP( "cream of celery soup" ), 6479}, + {I18N_NOOP( "cream of chicken soup" ), 6483}, + {I18N_NOOP( "cream of chicken soup, condensed" ), 6016}, + {I18N_NOOP( "cream of mushroom soup" ), 6443}, + {I18N_NOOP( "cream of mushroom soup, condensed" ), 6043}, + {I18N_NOOP( "croutons to serve" ), 18243}, + {I18N_NOOP( "crushed pineapple in juice" ), 9268}, + {I18N_NOOP( "cucumber" ), 11205}, + {I18N_NOOP( "cucumber, whole" ), 11206}, + {I18N_NOOP( "cucumbers" ), 11206}, + {I18N_NOOP( "cucumbers, whole" ), 11206}, + {I18N_NOOP( "cumin" ), 2014}, + {I18N_NOOP( "currants" ), 9084}, + {I18N_NOOP( "dehydrated onion flakes" ), 11284}, + {I18N_NOOP( "diced chicken" ), 5039}, + {I18N_NOOP( "dried basil" ), 2044}, + {I18N_NOOP( "dried onion" ), 11284}, + {I18N_NOOP( "dried oregano" ), 2027}, + {I18N_NOOP( "dried parsley" ), 2029}, + {I18N_NOOP( "dried porcini mushrooms" ), 11268}, + {I18N_NOOP( "dried red pepper" ), 2031}, + {I18N_NOOP( "dried sage" ), 2038}, + {I18N_NOOP( "dry bread crumbs" ), 18079}, + {I18N_NOOP( "dry mustard" ), 2024}, + {I18N_NOOP( "dry nonfat milk powder" ), 1091}, + {I18N_NOOP( "egg" ), 1123}, + {I18N_NOOP( "egg noodles (medium)" ), 20409}, + {I18N_NOOP( "egg substitute" ), 1144}, + {I18N_NOOP( "eggplant" ), 11209}, + {I18N_NOOP( "eggplant (baby)" ), 11209}, + {I18N_NOOP( "eggs" ), 1134}, + {I18N_NOOP( "eggs, hard cooked" ), 1129}, + {I18N_NOOP( "eggs, hard-boiled" ), 1129}, + {I18N_NOOP( "egg whites" ), 1124}, + {I18N_NOOP( "egg yolk" ), 1125}, + {I18N_NOOP( "egg yolks" ), 1125}, + {I18N_NOOP( "elbow macaroni" ), 20099}, + {I18N_NOOP( "flaked almonds" ), 12061}, + {I18N_NOOP( "flat anchovies" ), 15001}, + {I18N_NOOP( "flour" ), 20081}, + {I18N_NOOP( "flour tortillas" ), 18364}, + {I18N_NOOP( "flour, all-purpose" ), 20081}, + {I18N_NOOP( "flour, white" ), 20581}, + {I18N_NOOP( "flour, whole-grain wheat" ), 20080}, + {I18N_NOOP( "fresh spinach" ), 11457}, + {I18N_NOOP( "freshly chopped coriander" ), 11165}, + {I18N_NOOP( "freshly chopped parsley" ), 11297}, + {I18N_NOOP( "frozen broccoli" ), 11092}, + {I18N_NOOP( "frozen broccoli spears" ), 11094}, + {I18N_NOOP( "frozen cauliflower" ), 11137}, + {I18N_NOOP( "frozen corn" ), 11179}, + {I18N_NOOP( "frozen egg whites" ), 1172}, + {I18N_NOOP( "frozen fish" ), 15027}, + {I18N_NOOP( "frozen green beans" ), 11060}, + {I18N_NOOP( "frozen orange juice concentrate" ), 9214}, + {I18N_NOOP( "frozen peas" ), 11814}, + {I18N_NOOP( "frozen whole egg" ), 1171}, + {I18N_NOOP( "garlic" ), 11215}, + {I18N_NOOP( "garlic clove" ), 11215}, + {I18N_NOOP( "garlic cloves" ), 11215}, + {I18N_NOOP( "garlic powder" ), 2020}, + {I18N_NOOP( "garlic salt" ), 2020}, + {I18N_NOOP( "gelatin" ), 19173}, + {I18N_NOOP( "gelatin (peach-flavored)" ), 14397}, + {I18N_NOOP( "ginger" ), 2021}, + {I18N_NOOP( "ginger, fresh" ), 2021}, + {I18N_NOOP( "graham cracker crust (8 or 9 in)" ), 18173}, + {I18N_NOOP( "granny Smith apple" ), 9003}, + {I18N_NOOP( "granulated garlic" ), 16125}, + {I18N_NOOP( "grapefruit" ), 9112}, + {I18N_NOOP( "gravy" ), 6116}, + {I18N_NOOP( "green beans (fresh)" ), 11052}, + {I18N_NOOP( "green chile" ), 11980}, + {I18N_NOOP( "green chiles" ), 11670}, + {I18N_NOOP( "green chili pepper" ), 11670}, + {I18N_NOOP( "green onion" ), 11282}, + {I18N_NOOP( "green pepper" ), 11333}, + {I18N_NOOP( "green peppers" ), 11333}, + {I18N_NOOP( "ground almonds" ), 12061}, + {I18N_NOOP( "ground beef" ), 13309}, + {I18N_NOOP( "ground chuck" ), 13302}, + {I18N_NOOP( "ground cinnamon" ), 2010}, + {I18N_NOOP( "ground clove" ), 2011}, + {I18N_NOOP( "ground cloves" ), 2011}, + {I18N_NOOP( "ground cumin" ), 2014}, + {I18N_NOOP( "ground ginger" ), 2021}, + {I18N_NOOP( "ground nutmeg" ), 2025}, + {I18N_NOOP( "ground oregano" ), 2027}, + {I18N_NOOP( "ground pepper" ), 2030}, + {I18N_NOOP( "ground thyme" ), 2042}, + {I18N_NOOP( "ham" ), 7027}, + {I18N_NOOP( "hamburger" ), 13309}, + {I18N_NOOP( "hamburger bun" ), 18641}, + {I18N_NOOP( "hamburger buns" ), 18641}, + {I18N_NOOP( "hamburger rolls" ), 18641}, + {I18N_NOOP( "hazelnuts" ), 12120}, + {I18N_NOOP( "heavy cream" ), 1053}, + {I18N_NOOP( "herb stuffing" ), 18082}, + {I18N_NOOP( "honey" ), 19296}, + {I18N_NOOP( "horseradish" ), 2055}, + {I18N_NOOP( "hot pepper flakes" ), 2031}, + {I18N_NOOP( "hot pepper sauce" ), 6168}, + {I18N_NOOP( "hot sauce" ), 6164}, + {I18N_NOOP( "hot water" ), 14429}, + {I18N_NOOP( "instant dry milk" ), 1155}, + {I18N_NOOP( "instant rice" ), 20048}, + {I18N_NOOP( "Italian salad dressing" ), 4114}, + {I18N_NOOP( "jalapeno peppers" ), 11979}, + {I18N_NOOP( "juice of 1 lemon" ), 9152}, + {I18N_NOOP( "ketchup" ), 11935}, + {I18N_NOOP( "lasagna noodles (8 noodles)" ), 20409}, + {I18N_NOOP( "leaf lettuce" ), 11253}, + {I18N_NOOP( "leeks" ), 11246}, + {I18N_NOOP( "lemon" ), 9150}, + {I18N_NOOP( "lemon juice" ), 9152}, + {I18N_NOOP( "lemon juice, bottled" ), 9153}, + {I18N_NOOP( "lemon juice, frozen" ), 9154}, + {I18N_NOOP( "lemon peel" ), 9156}, + {I18N_NOOP( "lemon, sliced" ), 9151}, + {I18N_NOOP( "lentil" ), 16069}, + {I18N_NOOP( "lettuce" ), 11253}, + {I18N_NOOP( "lime juice" ), 9160}, + {I18N_NOOP( "long-grain white rice" ), 20044}, + {I18N_NOOP( "low sodium beef broth" ), 6475}, + {I18N_NOOP( "low sodium chicken broth" ), 6413}, + {I18N_NOOP( "lowfat 1% milk" ), 1084}, + {I18N_NOOP( "lowfat cheddar cheese" ), 1168}, + {I18N_NOOP( "lowfat cottage cheese" ), 1016}, + {I18N_NOOP( "lowfat mayonnaise" ), 4018}, + {I18N_NOOP( "lowfat mozzarella cheese" ), 1028}, + {I18N_NOOP( "lowfat yogurt" ), 1117}, + {I18N_NOOP( "low-sodium chicken broth" ), 6413}, + {I18N_NOOP( "low-sodium soy sauce" ), 16125}, + {I18N_NOOP( "low-sodium vegetable stock" ), 6468}, + {I18N_NOOP( "margarine" ), 4071}, + {I18N_NOOP( "margarine, soft" ), 4092}, + {I18N_NOOP( "marjoram" ), 2023}, + {I18N_NOOP( "marjoram leaves" ), 2023}, + {I18N_NOOP( "mayonnaise" ), 4025}, + {I18N_NOOP( "medium-size egg" ), 1123}, + {I18N_NOOP( "milk" ), 1081}, + {I18N_NOOP( "mint leaves" ), 2064}, + {I18N_NOOP( "mixed fruit in syrup" ), 9099}, + {I18N_NOOP( "mixed vegetables" ), 11318}, + {I18N_NOOP( "mixed vegetables, frozen" ), 11583}, + {I18N_NOOP( "molasses" ), 19304}, + {I18N_NOOP( "molasses, dark" ), 19305}, + {I18N_NOOP( "monterey jack cheese" ), 1025}, + {I18N_NOOP( "mozzarella cheese" ), 1028}, + {I18N_NOOP( "mung beans" ), 11626}, + {I18N_NOOP( "mushroom" ), 11260}, + {I18N_NOOP( "mushrooms" ), 11260}, + {I18N_NOOP( "mustard" ), 2046}, + {I18N_NOOP( "mustard seeds" ), 2024}, + {I18N_NOOP( "navy beans, cooked" ), 16338}, + {I18N_NOOP( "nonfat dry milk" ), 1092}, + {I18N_NOOP( "nonfat milk" ), 1085}, + {I18N_NOOP( "noodles" ), 20409}, + {I18N_NOOP( "noodles (lasagne)" ), 20409}, + {I18N_NOOP( "noodles (lasagne) (6-8 bunches)" ), 20409}, + {I18N_NOOP( "noodles, egg (medium)" ), 20409}, + {I18N_NOOP( "nutmeg" ), 2025}, + {I18N_NOOP( "nuts, chopped" ), 16087}, + {I18N_NOOP( "oil" ), 4518}, + {I18N_NOOP( "oil, peanut" ), 4042}, + {I18N_NOOP( "oil, sesame" ), 4058}, + {I18N_NOOP( "olive oil" ), 4053}, + {I18N_NOOP( "onion" ), 11282}, + {I18N_NOOP( "onion, medium" ), 11282}, + {I18N_NOOP( "onion powder" ), 2026}, + {I18N_NOOP( "onion salt" ), 2026}, + {I18N_NOOP( "onion soup" ), 6445}, + {I18N_NOOP( "onion, chopped" ), 11282}, + {I18N_NOOP( "onion, large" ), 11282}, + {I18N_NOOP( "onions" ), 11284}, + {I18N_NOOP( "onions, chopped" ), 11282}, + {I18N_NOOP( "orange juice" ), 9206}, + {I18N_NOOP( "orange juice, from frozen concentra" ), 9215}, + {I18N_NOOP( "orange rind" ), 9216}, + {I18N_NOOP( "orange zest" ), 9216}, + {I18N_NOOP( "oranges" ), 9200}, + {I18N_NOOP( "oregano" ), 2027}, + {I18N_NOOP( "Oreo Cookies" ), 18166}, + {I18N_NOOP( "paprika" ), 2028}, + {I18N_NOOP( "paprika pepper" ), 2028}, + {I18N_NOOP( "parmesan cheese" ), 1032}, + {I18N_NOOP( "parsley" ), 2029}, + {I18N_NOOP( "parsley flakes" ), 2029}, + {I18N_NOOP( "parsley stalks" ), 11297}, + {I18N_NOOP( "parsnips" ), 11298}, + {I18N_NOOP( "pasta shells" ), 20100}, + {I18N_NOOP( "peach slices in syrup" ), 9240}, + {I18N_NOOP( "peanut butter" ), 16098}, + {I18N_NOOP( "pearled barley" ), 20005}, + {I18N_NOOP( "peas, canned" ), 11308}, + {I18N_NOOP( "peas, frozen" ), 11312}, + {I18N_NOOP( "pecans" ), 12142}, + {I18N_NOOP( "pepper" ), 2030}, + {I18N_NOOP( "Pepperidge Farm stuffing" ), 18082}, + {I18N_NOOP( "pickle relish" ), 11945}, + {I18N_NOOP( "pickles" ), 11937}, + {I18N_NOOP( "pimento" ), 11943}, + {I18N_NOOP( "pine kernels" ), 12149}, + {I18N_NOOP( "pineapple chunks in juice" ), 9268}, + {I18N_NOOP( "pineapple juice" ), 9273}, + {I18N_NOOP( "pinto beans" ), 16042}, + {I18N_NOOP( "pinto beans, canned" ), 16044}, + {I18N_NOOP( "plain flour" ), 20081}, + {I18N_NOOP( "plain low-fat yogurt" ), 1117}, + {I18N_NOOP( "plain yogurt" ), 1117}, + {I18N_NOOP( "pork roast" ), 10083}, + {I18N_NOOP( "potato" ), 11352}, + {I18N_NOOP( "potato flakes" ), 11379}, + {I18N_NOOP( "potato Granules" ), 11380}, + {I18N_NOOP( "potato, canned" ), 11376}, + {I18N_NOOP( "potatoes" ), 11352}, + {I18N_NOOP( "potatoes (red-skinned)" ), 11352}, + {I18N_NOOP( "poultry seasoning" ), 2034}, + {I18N_NOOP( "powdered sugar" ), 19335}, + {I18N_NOOP( "processed American cheese" ), 1149}, + {I18N_NOOP( "provolone cheese" ), 1035}, + {I18N_NOOP( "prunes" ), 9291}, + {I18N_NOOP( "pumpkin" ), 11422}, + {I18N_NOOP( "radishes" ), 11429}, + {I18N_NOOP( "raisins" ), 9298}, + {I18N_NOOP( "red burgundy wine" ), 14096}, + {I18N_NOOP( "red onion" ), 11282}, + {I18N_NOOP( "red pepper" ), 2031}, + {I18N_NOOP( "red snapper fillets" ), 15101}, + {I18N_NOOP( "red wine" ), 14096}, + {I18N_NOOP( "reduced calorie mayonnaise" ), 4013}, + {I18N_NOOP( "relish" ), 11945}, + {I18N_NOOP( "rice" ), 20045}, + {I18N_NOOP( "ricotta cheese" ), 1037}, + {I18N_NOOP( "ripe dessert pears" ), 9252}, + {I18N_NOOP( "rolled oats" ), 20038}, + {I18N_NOOP( "rosemary" ), 2063}, + {I18N_NOOP( "rum flavoring or vanilla" ), 2050}, + {I18N_NOOP( "sage" ), 2038}, + {I18N_NOOP( "salad dressing (Miracle Whip)" ), 4025}, + {I18N_NOOP( "salad onions" ), 11282}, + {I18N_NOOP( "salsa" ), 6164}, + {I18N_NOOP( "salt" ), 2047}, + {I18N_NOOP( "sandwich rolls" ), 18349}, + {I18N_NOOP( "sausage" ), 7063}, + {I18N_NOOP( "scallions" ), 11291}, + {I18N_NOOP( "self-raising flour" ), 20082}, + {I18N_NOOP( "sesame oil" ), 4058}, + {I18N_NOOP( "shallots" ), 11677}, + {I18N_NOOP( "shortening" ), 4547}, + {I18N_NOOP( "shredded carrots" ), 11124}, + {I18N_NOOP( "shredded lettuce" ), 11252}, + {I18N_NOOP( "shrimp (raw, medium-size)" ), 15149}, + {I18N_NOOP( "skim milk" ), 1085}, + {I18N_NOOP( "skinless boneless chicken breast" ), 5062}, + {I18N_NOOP( "sliced carrots" ), 11124}, + {I18N_NOOP( "small mushrooms" ), 11260}, + {I18N_NOOP( "small onions or shallots" ), 11282}, + {I18N_NOOP( "smoked bacon" ), 10124}, + {I18N_NOOP( "soft bread crumbs" ), 18079}, + {I18N_NOOP( "sour cream" ), 1056}, + {I18N_NOOP( "soy sauce" ), 16125}, + {I18N_NOOP( "spaghetti" ), 20120}, + {I18N_NOOP( "spinach leaf" ), 11457}, + {I18N_NOOP( "sprig fresh thyme" ), 2049}, + {I18N_NOOP( "stew beef" ), 13286}, + {I18N_NOOP( "stock" ), 6413}, + {I18N_NOOP( "sugar" ), 19335}, + {I18N_NOOP( "sweet pickle" ), 11940}, + {I18N_NOOP( "sweet pickle relish" ), 11945}, + {I18N_NOOP( "sweet potato" ), 11647}, + {I18N_NOOP( "sweet potato, canned" ), 11645}, + {I18N_NOOP( "swiss cheese" ), 1040}, + {I18N_NOOP( "taco sauce" ), 6168}, + {I18N_NOOP( "taco seasoning mix" ), 2034}, + {I18N_NOOP( "taco shells" ), 18360}, + {I18N_NOOP( "taco spice" ), 2034}, + {I18N_NOOP( "thyme" ), 2042}, + {I18N_NOOP( "tomato juice" ), 11540}, + {I18N_NOOP( "tomato paste" ), 11887}, + {I18N_NOOP( "tomato sauce" ), 11549}, + {I18N_NOOP( "tomatoes" ), 11529}, + {I18N_NOOP( "tomatoes, canned" ), 11533}, + {I18N_NOOP( "tomatoes, stewed" ), 11693}, + {I18N_NOOP( "tortilla chips" ), 19056}, + {I18N_NOOP( "tostada shell" ), 18363}, + {I18N_NOOP( "tuna" ), 15126}, + {I18N_NOOP( "tuna in water, canned" ), 15126}, + {I18N_NOOP( "turkey" ), 7079}, + {I18N_NOOP( "turkey ham" ), 7264}, + {I18N_NOOP( "turmeric" ), 2043}, + {I18N_NOOP( "turnip" ), 11564}, + {I18N_NOOP( "unflavored gelatin" ), 19173}, + {I18N_NOOP( "vanilla" ), 2050}, + {I18N_NOOP( "vanilla extract" ), 2050}, + {I18N_NOOP( "vanilla pudding mix (instant)" ), 19202}, + {I18N_NOOP( "veal shank" ), 17278}, + {I18N_NOOP( "vegetable oil" ), 4518}, + {I18N_NOOP( "vegetable stock" ), 6468}, + {I18N_NOOP( "Velveeta" ), 1149}, + {I18N_NOOP( "vinegar" ), 2048}, + {I18N_NOOP( "walnuts" ), 12154}, + {I18N_NOOP( "warm water" ), 14429}, + {I18N_NOOP( "water" ), 14429}, + {I18N_NOOP( "whipped topping" ), 1054}, + {I18N_NOOP( "white flour" ), 20481}, + {I18N_NOOP( "white pepper" ), 2032}, + {I18N_NOOP( "white rice" ), 20050}, + {I18N_NOOP( "white turnips" ), 11564}, + {I18N_NOOP( "white vinegar" ), 2048}, + {I18N_NOOP( "white wine" ), 14106}, + {I18N_NOOP( "whole wheat flour" ), 20080}, + {I18N_NOOP( "yeast" ), 18375}, + {I18N_NOOP( "yellow squash" ), 11641}, + {I18N_NOOP( "zucchini" ), 11477}, + {I18N_NOOP( "zucchini slices" ), 11953}, + {0, 0} + }; + +#endif //USDA_INGREDIENT_DATA_H diff --git a/src/backends/usda_property_data.h b/src/backends/usda_property_data.h new file mode 100644 index 0000000..de00155 --- /dev/null +++ b/src/backends/usda_property_data.h @@ -0,0 +1,76 @@ +/*************************************************************************** +* Copyright (C) 2003 * +* * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef USDA_PROPERTY_DATA_H +#define USDA_PROPERTY_DATA_H + +#include <tdelocale.h> + +struct property_data +{ + int id; + TQString name; + const char *unit; +}; + +#define TOTAL_USDA_PROPERTIES 43 + +//NOTE: the following must be in this order +static property_data property_data_list[] = { + { -1, I18N_NOOP( "water" ), "g"}, + { -1, I18N_NOOP( "energy" ), "kcal"}, + { -1, I18N_NOOP( "protein" ), "g"}, + { -1, I18N_NOOP( "fat" ), "g"}, + { -1, I18N_NOOP( "ash" ), "g"}, + { -1, I18N_NOOP( "carbohydrates" ), "g"}, + { -1, I18N_NOOP( "dietary fiber" ), "g"}, + { -1, I18N_NOOP( "sugar" ), "g"}, + { -1, I18N_NOOP( "calcuim" ), "mg"}, + { -1, I18N_NOOP( "iron" ), "mg"}, + { -1, I18N_NOOP( "magnesium" ), "mg"}, + { -1, I18N_NOOP( "phosphorus" ), "mg"}, + { -1, I18N_NOOP( "potassium" ), "mg"}, + { -1, I18N_NOOP( "sodium" ), "mg"}, + { -1, I18N_NOOP( "zinc" ), "mg"}, + { -1, I18N_NOOP( "copper" ), "mg"}, + { -1, I18N_NOOP( "manganese" ), "mg"}, + { -1, I18N_NOOP( "selenium" ), "g"}, + { -1, I18N_NOOP( "vitamin C" ), "mg"}, + { -1, I18N_NOOP( "thiamin" ), "mg"}, + { -1, I18N_NOOP( "riboflavin" ), "mg"}, + { -1, I18N_NOOP( "niacin" ), "mg"}, + { -1, I18N_NOOP( "pantothenic acid" ), "mg"}, + { -1, I18N_NOOP( "vitamin B" ), "mg"}, + { -1, I18N_NOOP( "folate" ), "g"}, + { -1, I18N_NOOP( "folic acid" ), "g"}, + { -1, I18N_NOOP( "food folate" ), "g"}, + { -1, I18N_NOOP( "folate (DFE)" ), "g"}, + { -1, I18N_NOOP( "vitamin B12" ), "g"}, + { -1, I18N_NOOP( "vitamin A" ), "IU"}, + { -1, I18N_NOOP( "vitamin A (RAE)" ), "mg"}, + { -1, I18N_NOOP( "retinol" ), "g"}, + { -1, I18N_NOOP( "vitamin E" ), "g"}, + { -1, I18N_NOOP( "vitamin K" ), "g"}, + { -1, I18N_NOOP( "alpha-carotene" ), "g"}, + { -1, I18N_NOOP( "beta-carotene" ), "g"}, + { -1, I18N_NOOP( "beta-cryptoxanthin" ), "g"}, + { -1, I18N_NOOP( "lycopene" ), "g"}, + { -1, I18N_NOOP( "lutein+zeazanthin" ), "g"}, + { -1, I18N_NOOP( "saturated fat" ), "g"}, + { -1, I18N_NOOP( "monounsaturated fat" ), "g"}, + { -1, I18N_NOOP( "polyunsaturated fat" ), "g"}, + { -1, I18N_NOOP( "cholesterol" ), "mg"}, + { 0, 0, 0 } + }; + +#endif //USDA_PROPERTY_DATA_H diff --git a/src/backends/usda_unit_data.h b/src/backends/usda_unit_data.h new file mode 100644 index 0000000..d509004 --- /dev/null +++ b/src/backends/usda_unit_data.h @@ -0,0 +1,104 @@ +/*************************************************************************** +* Copyright (C) 2006 * +* * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef USDA_UNIT_DATA_H +#define USDA_UNIT_DATA_H + +#include <tdelocale.h> + +#include <tqstring.h> + +struct unit_data { + const char *name; + const char *plural; +}; + +static unit_data unit_data_list[] = { + {"bag","bags"}, + {"block","blocks"}, + {"bottle","bottles"}, + {"box","boxes"}, + {"bunch","bunches"}, + {"can","cans"}, + {"cone","cones"}, + {"container","containers"}, + {"cube","cubes"}, + {"cup","cups"}, + {"fl oz","fl oz"}, + {"glass","glasses"}, + {"item","items"}, + {"loaf","loaves"}, + {"large","large"}, + {"lb","lbs"}, + {"junior","junior"}, + {"leaf","leaves"}, + {"medium","medium"}, + {"oz","oz"}, + {"pack","packs"}, + {"package","packages"}, + {"packet","packets"}, + {"piece","pieces"}, + {"pouch","pouches"}, + {"quart","quarts"}, + {"scoop","scoops"}, + {"sheet","sheets"}, + {"slice","slices"}, + {"small","small"}, + {"spear","spears"}, + {"sprout","spouts"}, + {"sprig","sprigs"}, + {"square","squares"}, + {"stalk","stalks"}, + {"stem","stems"}, + {"strip","strips"}, + {"tablespoon","tablespoons"}, + {"tbsp","tbsp"}, + {"teaspoon","teaspoons"}, + {"tsp","tsp"}, + {"tube","tubes"}, + {"unit","units"}, + {0,0} +}; + +static const char * prep_data_list[] = { + "chopped", + "diced", + "sliced", + "crumbled", + "crushed", + "ground", + "grated", + "mashed", + "melted", + "packed", + "pureed", + "quartered", + "thawed", + "shredded", + "sifted", + "pared", + "flaked", + "unpacked", + "unsifted", + "unthawed", + "pitted", + "peeled", + "cooked", + "hulled", + "shelled", + "raw", + "whipped", + 0 +}; + +bool parseUSDAUnitAndPrep( const TQString &string, TQString &unit, TQString &prep ); + +#endif //USDA_UNIT_DATA_H diff --git a/src/configure.in.bot b/src/configure.in.bot new file mode 100644 index 0000000..6116164 --- /dev/null +++ b/src/configure.in.bot @@ -0,0 +1,57 @@ +echo "" +echo "" +echo "----- Configure Results -----" +echo "- -" + +if test "x$have_mysql" = "xtrue"; then +echo "- MySQL Support................... YES -" +else +echo "- MySQL Support................... NO -" +fi + +echo "- -" + +if test "x$have_postgresql" = "xtrue"; then +echo "- PostgreSQL Support.............. YES -" +else +echo "- PostgreSQL Support.............. NO -" +fi + +echo "- -" + +if test "x$have_sqlite" = "xtrue" || test "x$have_sqlite3" = "xtrue"; then +echo "- SQLite Found..................... YES -" +else +echo "- SQLite Found..................... NO -" +fi +echo "- -" +echo "------------------------------------------------" + + +if test "x$will_not_build_krecipes" = "xtrue"; then +echo "" +echo "*** Krecipes needs a database backend enabled. If ***" +echo "*** you wish to use SQLite, you must have SQLite ***" +echo "*** installed before compiling. ***" +echo "*** KRECIPES WILL NOT BE BUILT ***" +echo "" +echo "You can get SQLite from: http://www.hwaci.com/sw/sqlite/" +echo "Or MySQL from: http://www.mysql.com" + +else if test "x$have_mysql" = "xtrue"; then + if test "x$have_sqlite" = "xtrue" || test "x$have_sqlite3" = "xtrue"; then + echo "- -" + echo "- Fine, you can build Krecipes now -" + echo "------------------------------------------------" + else + echo "- -" + echo "- You don't have SQLite installed (or have -" + echo "- chosen to disable it), but Krecipes can -" + echo "- work just fine with another supported -" + echo "- database. -" + echo "- If you still want to, you can get SQLite -" + echo "- from http://www.hwaci.com/sw/sqlite/ -" + echo "------------------------------------------------" + fi + fi +fi diff --git a/src/configure.in.in b/src/configure.in.in new file mode 100644 index 0000000..530b8f1 --- /dev/null +++ b/src/configure.in.in @@ -0,0 +1,96 @@ +#MIN_CONFIG(3.1) + +AM_INIT_AUTOMAKE(krecipes, 0.8) + +#KDE_USE_TQT(3.1) + +AC_C_BIGENDIAN +AC_CHECK_KDEMAXPATHLEN + +AC_ARG_WITH(sqlite, AC_HELP_STRING([--without-sqlite], [Don't compile SQLite backend support]), , with_sqlite=yes) +AC_ARG_WITH(mysql, AC_HELP_STRING([--without-mysql], [Don't compile MySQL backend support]), , with_mysql=yes) +AC_ARG_WITH(postgresql, AC_HELP_STRING([--without-postgresql], [Don't compile PostgreSQL backend support]), , with_postgresql=yes) + +dnl ----- Check if we should enable MySQL ------ + +if test "x$with_mysql" != "xno"; then + AC_DEFINE(HAVE_MYSQL, 1, [have MySQL]) + have_mysql="true" +else + AC_DEFINE(HAVE_MYSQL, 0, [have MySQL]) + have_mysql="false" +fi + + +dnl ----- Check if we should enable PostgreSQL ------ + +if test "x$with_postgresql" != "xno"; then + AC_DEFINE(HAVE_POSTGRESQL, 1, [have PostgreSQL]) + have_postgresql="true" +else + AC_DEFINE(HAVE_POSTGRESQL, 0, [have PostgreSQL]) + have_postgresql="false" +fi + + +dnl ------ Check for the SQLite headers ----- + +AC_DEFUN([AC_HAVE_SQLITE], +[ + AC_DEFINE(HAVE_SQLITE, 1, [have SQLite]) + have_sqlite=true + SQLITE_LIB="-lsqlite" + AC_SUBST(SQLITE_LIB) +]) + +AC_DEFUN([AC_NO_SQLITE], +[ + AC_DEFINE(HAVE_SQLITE, 0, [have SQLite]) + have_sqlite=false +]) + +AC_DEFUN([AC_HAVE_SQLITE3], +[ + AC_DEFINE(HAVE_SQLITE3, 1, [have SQLite3]) + have_sqlite3=true + AC_NO_SQLITE + SQLITE_LIB="-lsqlite3" + AC_SUBST(SQLITE_LIB) +]) + +AC_DEFUN([AC_NO_SQLITE3], +[ + AC_DEFINE(HAVE_SQLITE3, 0, [have SQLite3]) + have_sqlite3=false +]) + +if test "x$with_sqlite" != "xno"; then + KDE_CHECK_HEADER(sqlite3.h, + AC_HAVE_SQLITE3, + AC_NO_SQLITE3 + ) + + if test "x$have_sqlite3" = "xfalse"; then + KDE_CHECK_HEADER(sqlite.h, + AC_HAVE_SQLITE, + AC_NO_SQLITE + ) + fi + +else + AC_NO_SQLITE + AC_NO_SQLITE3 +fi + +dnl ----------- Should I link or should I not link? --------- +AM_CONDITIONAL(link_lib_MYSQL, test x$have_mysql = xtrue) +AM_CONDITIONAL(link_lib_POSTGRESQL, test x$have_postgresql = xtrue) +AM_CONDITIONAL(link_lib_SQLITE, test x$have_sqlite = xtrue || test x$have_sqlite3 = xtrue) + +dnl ----------- Should Krecipes be built at all? ------------ +if test "x$have_mysql" = "xfalse" && test "x$have_postgresql" = "xfalse"; then + if test "x$have_sqlite" = "xfalse" && test "x$have_sqlite3" = "xfalse"; then + DO_NOT_COMPILE="$DO_NOT_COMPILE krecipes" + will_not_build_krecipes=true + fi +fi diff --git a/src/convert_sqlite3.cpp b/src/convert_sqlite3.cpp new file mode 100644 index 0000000..ddb144e --- /dev/null +++ b/src/convert_sqlite3.cpp @@ -0,0 +1,121 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected] * +* * +* 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. * +***************************************************************************/ + +#include "convert_sqlite3.h" + +#include <tqapplication.h> +#include <tqfile.h> + +#include <kdebug.h> +#include <tdeglobal.h> +#include <tdeconfig.h> +#include <tdemessagebox.h> +#include <kprocio.h> + +//FIXME: Some messages should be given to the user about success/failure, but that can't be done in the 0.8.x branch due to i18n. + +ConvertSQLite3::ConvertSQLite3( const TQString &db_file ) : TQObject(), error(false) +{ + TQString file = db_file; + if ( file.isEmpty() ) { + TDEConfig *config = TDEGlobal::config(); + config->setGroup("Server"); + file = config->readEntry("DBFile"); + } + + KProcIO *p = new KProcIO; + p->setUseShell(true); + + //sqlite OLD.DB .dump | sqlite3 NEW.DB + *p << "sqlite" << file << ".dump" << + "|" << "sqlite3" << file+".new"; + + TQApplication::connect( p, TQ_SIGNAL(readReady(KProcIO*)), this, TQ_SLOT(processOutput(KProcIO*)) ); + + bool success = p->start( TDEProcess::Block, true ); + if ( !success ) { + kdDebug()<<"Conversion failed... unable to start TDEProcess"<<endl; + return; + } + + if ( error ) + return; + + + TQString backup_file = file+".sqlite2"; + int i = 1; + while ( TQFile::exists(backup_file) ) { + backup_file = backup_file.left(file.length()+8)+"."+TQString::number(i); + ++i; + } + + if ( !copyFile( file, backup_file ) ) { + kdDebug()<<"Unable to backup SQLite 2 database... aborting"<<endl + <<"A successfully converted SQLite 3 file is available at \""<<file<<".new\"."<<endl; + } + else { + kdDebug()<<"SQLite 2 database backed up to "<<backup_file<<endl; + if ( !copyFile( file+".new", file ) ) { + kdDebug()<<"Unable to copy the new SQLite 3 database to: "<<file<<"."<<endl + <<"You may manually move \""<<file<<".new\" to \""<<file<<"\""<<endl; + } + else { + kdDebug()<<"Conversion successful!"<<endl; + TQFile::remove(file+".new"); + } + } +} + +ConvertSQLite3::~ConvertSQLite3() +{ +} + +void ConvertSQLite3::processOutput( KProcIO* p ) +{ + TQString error_str, buffer; + while ( p->readln(buffer) != -1 ) { + error_str += buffer; + } + + KMessageBox::error( 0, error_str ); + + error = true; + + p->ackRead(); +} + +bool ConvertSQLite3::copyFile( const TQString &oldFilePath, const TQString &newFilePath ) +{ + //load both files + TQFile oldFile(oldFilePath); + TQFile newFile(newFilePath); + bool openOld = oldFile.open( IO_ReadOnly ); + bool openNew = newFile.open( IO_WriteOnly ); + + //if either file fails to open bail + if(!openOld || !openNew) { return false; } + + //copy contents + uint BUFFER_SIZE = 16000; + char* buffer = new char[BUFFER_SIZE]; + while(!oldFile.atEnd()) + { + TQ_ULONG len = oldFile.readBlock( buffer, BUFFER_SIZE ); + newFile.writeBlock( buffer, len ); + } + + //deallocate buffer + delete[] buffer; + buffer = NULL; + return true; +} + +#include "convert_sqlite3.moc" + diff --git a/src/convert_sqlite3.h b/src/convert_sqlite3.h new file mode 100644 index 0000000..4f23bf5 --- /dev/null +++ b/src/convert_sqlite3.h @@ -0,0 +1,35 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected] * +* * +* 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. * +***************************************************************************/ + +#ifndef CONVERT_SQLITE3_H +#define CONVERT_SQLITE3_H + +#include <tqobject.h> + +class KProcIO; + +class ConvertSQLite3 : public TQObject +{ +TQ_OBJECT + +public: + ConvertSQLite3( const TQString &db_file = TQString::null ); + ~ConvertSQLite3(); + +public slots: + void processOutput( KProcIO* p ); + +private: + bool copyFile( const TQString &oldFilePath, const TQString &newFilePath ); + + bool error; +}; + +#endif //CONVERT_SQLITE3_H diff --git a/src/datablocks/Makefile.am b/src/datablocks/Makefile.am new file mode 100644 index 0000000..e6161f7 --- /dev/null +++ b/src/datablocks/Makefile.am @@ -0,0 +1,20 @@ +## Makefile.am for krecipes + +# this is the program that gets installed. it's name is used for all +# of the other Makefile.am variables + +# set the include path for X, tqt and TDE +INCLUDES = -I$(srcdir)/.. $(all_includes) + +noinst_LTLIBRARIES=libdatablocks.la +libdatablocks_la_SOURCES= \ + recipelist.cpp constraintlist.cpp categorytree.cpp kreborder.cpp \ + recipe.cpp ingredient.cpp ingredientlist.cpp elementlist.cpp \ + element.cpp ingredientproperty.cpp ingredientpropertylist.cpp \ + unit.cpp unitratio.cpp unitratiolist.cpp mixednumber.cpp rating.cpp \ + weight.cpp + +libdatablocks_la_METASOURCES=AUTO + +#the library search path. +libdatablocks_la_LDFLAGS = $(KDE_RPATH) $(all_libraries) diff --git a/src/datablocks/categorytree.cpp b/src/datablocks/categorytree.cpp new file mode 100644 index 0000000..bb61de6 --- /dev/null +++ b/src/datablocks/categorytree.cpp @@ -0,0 +1,133 @@ +/*************************************************************************** +* Copyright (C) 2004 by Jason Kivlighn * +* ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "categorytree.h" + +#include <kdebug.h> + +#include "element.h" + +CategoryTree::CategoryTree( CategoryTree *parent ) : + m_parent( 0 ), m_child( 0 ), m_sibling( 0 ), m_last(0), m_count(0) +{ + if ( parent ) + parent->insertItem( this ); +} + +CategoryTree::~CategoryTree() +{ + if ( m_parent ) + m_parent->takeItem( this ); + + CategoryTree * i = m_child; + m_child = 0; + while ( i ) { + i->m_parent = 0; + CategoryTree * n = i->m_sibling; + delete i; + i = n; + } +} + +CategoryTree *CategoryTree::add + ( const Element &cat ) +{ + CategoryTree * new_child = new CategoryTree( this ); + new_child->category = cat; + + m_count++; + + return new_child; +} + +void CategoryTree::insertItem( CategoryTree *newChild ) +{ + newChild->m_parent = this; + if ( m_child && m_child->m_last ) + m_child->m_last->m_sibling = newChild; + else + m_child = newChild; + + m_child->m_last = newChild; + m_count++; +} + +void CategoryTree::takeItem( CategoryTree *tree ) +{ + kdError()<< "Both these methods seem to be broken... don't use this function!" << endl; + + CategoryTree *lastItem = m_child->m_last; +#if 0 + CategoryTree ** nextChild = &m_child; + while( nextChild && *nextChild && tree != *nextChild ) + nextChild = &((*nextChild)->m_sibling); + + if ( nextChild && tree == *nextChild ) { + *nextChild = (*nextChild)->m_sibling; + } + tree->m_parent = 0; + tree->m_sibling = 0; +#else + for ( CategoryTree *it = firstChild(); it; it = it->nextSibling() ) { + if ( it->nextSibling() == tree ) { + it->m_sibling = tree->nextSibling(); + break; + } + } + tree->m_parent = 0; + tree->m_sibling = 0; +#endif + if ( tree != m_last ) + m_child->m_last = lastItem; + else //FIXME: unstable behavior if this is the case + kdDebug()<<"CategoryTree::takeItem: warning - unstable behavior expected"<<endl; + m_count--; +} + +void CategoryTree::clear() +{ + CategoryTree *c = m_child; + CategoryTree *n; + while( c ) { + n = c->m_sibling; + delete c; + c = n; + } +} + +bool CategoryTree::contains( int id ) const +{ + bool result = false; + + for ( CategoryTree * child_it = firstChild(); child_it; child_it = child_it->nextSibling() ) { + if ( child_it->category.id == id ) + return true; + + result = child_it->contains( id ); + } + + return result; +} + +CategoryTree* CategoryTree::find( int id ) const +{ + CategoryTree *result = 0; + + for ( CategoryTree * child_it = firstChild(); child_it; child_it = child_it->nextSibling() ) { + if ( child_it->category.id == id ) + return child_it; + + result = child_it->find( id ); + } + + return result; +} + + diff --git a/src/datablocks/categorytree.h b/src/datablocks/categorytree.h new file mode 100644 index 0000000..e5668e4 --- /dev/null +++ b/src/datablocks/categorytree.h @@ -0,0 +1,61 @@ +/*************************************************************************** +* Copyright (C) 2004 by Jason Kivlighn * +* ([email protected]) * +* * +* 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. * +***************************************************************************/ +#ifndef CATEGORYTREE_H +#define CATEGORYTREE_H + +#include "element.h" + +class CategoryTree +{ +public: + CategoryTree( CategoryTree *parent = 0 ); + + ~CategoryTree(); + + Element category; + + CategoryTree *add + ( const Element &cat ); + void clear(); + + bool contains( int id ) const; + CategoryTree* find( int id ) const; + + CategoryTree *parent() const + { + return m_parent; + } + CategoryTree *firstChild() const + { + return m_child; + } + CategoryTree *nextSibling() const + { + return m_sibling; + } + + void takeItem( CategoryTree * ); + void insertItem( CategoryTree * ); + + int count() const { return m_count; } + +private: + CategoryTree( const CategoryTree & ); + CategoryTree &operator=( const CategoryTree & ); + + CategoryTree *m_parent; + CategoryTree *m_child; + CategoryTree *m_sibling; + CategoryTree *m_last; + + int m_count; +}; + +#endif diff --git a/src/datablocks/constraintlist.cpp b/src/datablocks/constraintlist.cpp new file mode 100644 index 0000000..f96ffc9 --- /dev/null +++ b/src/datablocks/constraintlist.cpp @@ -0,0 +1,11 @@ +/*************************************************************************** +* Copyright (C) 2003 by Unai Garro * +* [email protected] * +* * +* 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. * +***************************************************************************/ +#include "constraintlist.h" + diff --git a/src/datablocks/constraintlist.h b/src/datablocks/constraintlist.h new file mode 100644 index 0000000..2a75470 --- /dev/null +++ b/src/datablocks/constraintlist.h @@ -0,0 +1,47 @@ +/*************************************************************************** +* Copyright (C) 2003 by Unai Garro * +* [email protected] * +* * +* 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. * +***************************************************************************/ +#ifndef CONSTRAINTLIST_H +#define CONSTRAINTLIST_H + +#include <tqstring.h> + +#include "unit.h" + +/** +@author Unai Garro +*/ + +class IngredientProperty; + +class Constraint +{ +public: + Constraint() + { + max = 0.0; + min = 0.0; + id = -1; + enabled = false; + } + + ~Constraint(){} + + int id; + TQString name; + TQString units; + Unit perUnit; // stores the unit ID, name, and type + double max; + double min; + bool enabled; +}; + +typedef TQValueList< Constraint > ConstraintList; + +#endif diff --git a/src/datablocks/element.cpp b/src/datablocks/element.cpp new file mode 100644 index 0000000..18c2b7f --- /dev/null +++ b/src/datablocks/element.cpp @@ -0,0 +1,43 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "element.h" + +Element::Element() : + id( -1 ) +{} + +Element::Element( const TQString &_name, int _id ) : + name( _name ), + id( _id ) +{} + +Element::Element( const Element &el ) +{ + id = el.id; + name = el.name; +} + +Element::~Element() +{} + +Element& Element::operator=( const Element &el ) +{ + id = el.id; + name = el.name; + return *this; +} + +bool Element::operator==( const Element &el ) const +{ + return ( el.id == id ); +} diff --git a/src/datablocks/element.h b/src/datablocks/element.h new file mode 100644 index 0000000..6324719 --- /dev/null +++ b/src/datablocks/element.h @@ -0,0 +1,41 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef ELEMENT_H +#define ELEMENT_H +#include <tqstring.h> +/** +@author Unai Garro +*/ +class Element +{ +public: + Element(); + Element( const TQString &name, int id = -1 ); + Element( const Element &el ); + + ~Element(); + TQString name; + int id; + Element& operator=( const Element &el ); + + /** Compare two elements by their id */ + bool operator==( const Element & ) const; + + /** Compare (sort) by name */ + bool operator<( const Element &e ) const + { + return name < e.name; + } +}; + +#endif diff --git a/src/datablocks/elementlist.cpp b/src/datablocks/elementlist.cpp new file mode 100644 index 0000000..7eca42d --- /dev/null +++ b/src/datablocks/elementlist.cpp @@ -0,0 +1,102 @@ +/*************************************************************************** +* Copyright (C) 2003-2004 by * +* Unai Garro ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ +#include "datablocks/elementlist.h" + +ElementList::ElementList() : TQValueList <Element>() +{} + +ElementList::~ElementList() +{} + +Element ElementList::getElement( int index ) const +{ + return * ( at( index ) ); +} + +Element ElementList::findByName( const TQString &name ) const +{ + ElementList::const_iterator it_end = end(); + for ( ElementList::const_iterator it = begin(); it != it_end; ++it ) { + if ( ( *it ).name == name ) + return * it; + } + + Element el; + el.id = -1; + return el; +} + +Element ElementList::findByName( const TQRegExp &rx ) const +{ + ElementList::const_iterator it_end = end(); + for ( ElementList::const_iterator it = begin(); it != it_end; ++it ) { + if ( ( *it ).name.find(rx) != -1 ) + return * it; + } + + Element el; + el.id = -1; + return el; +} + +bool ElementList::containsId( int id ) const // Search by id (which uses search by item, with comparison defined on header) +{ + if ( id == -1 ) { + return count() == 0; + } + + Element i; + i.id = id; + return ( find( i ) != end() ); +} + +bool ElementList::containsSubSet( ElementList &el ) +{ + ElementList::const_iterator it_end = el.end(); + ElementList::const_iterator it; + + for ( it = el.begin(); it != it_end; ++it ) { + if ( !containsId( ( *it ).id ) ) + return false; + } + return true; +} + +TQString ElementList::join( const TQString &sep ) const +{ + TQString ret; + + ElementList::const_iterator it_end = end(); + ElementList::const_iterator it; + + for ( it = begin(); it != it_end; ++it ) { + if ( it != begin() ) + ret += sep; + ret += (*it).name; + } + + return ret; +} + +ElementList ElementList::split( const TQString &sep, const TQString &str ) +{ + ElementList ret; + TQStringList list = TQStringList::split(sep,str); + + TQStringList::const_iterator it_end = list.end(); + TQStringList::const_iterator it; + + for ( it = list.begin(); it != it_end; ++it ) { + ret.append( Element((*it).stripWhiteSpace()) ); + } + + return ret; +} diff --git a/src/datablocks/elementlist.h b/src/datablocks/elementlist.h new file mode 100644 index 0000000..58d5bca --- /dev/null +++ b/src/datablocks/elementlist.h @@ -0,0 +1,43 @@ +/*************************************************************************** +* Copyright (C) 2003-2004 by * +* Unai Garro ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ +#ifndef ELEMENTLIST_H +#define ELEMENTLIST_H + +#include <tqvaluelist.h> +#include <tqregexp.h> + +#include "element.h" + +/** +@author Unai Garro +*/ +class ElementList: public TQValueList<Element> +{ +public: + ElementList(); + ~ElementList(); + + bool containsId( int id ) const; + bool containsSubSet( ElementList &el ); + + Element findByName( const TQString & ) const; + Element findByName( const TQRegExp & ) const; + + Element getElement( int index ) const; + + TQString join( const TQString &sep ) const; + + static ElementList split( const TQString &sep, const TQString &str ); +}; + +typedef TQValueList<int> IDList; + +#endif diff --git a/src/datablocks/ingredient.cpp b/src/datablocks/ingredient.cpp new file mode 100644 index 0000000..a17beda --- /dev/null +++ b/src/datablocks/ingredient.cpp @@ -0,0 +1,95 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "datablocks/ingredient.h" + +#include <tqstringlist.h> + +#include "mixednumber.h" + +IngredientData::IngredientData() : amount( 0 ), amount_offset( 0 ), groupID( -1 ) +{} + +IngredientData::IngredientData( const TQString &_name, double _amount, const Unit &_units, int _unitID, int _ingredientID ) : + ingredientID( _ingredientID ), + name( _name ), + amount( _amount ), + amount_offset( 0 ), + units( _units ), + groupID( -1 ) +{ +units.id=_unitID; +} + +//compare also using the group id because there may be the same ingredient in a list multiple times, but each in a different group +bool IngredientData::operator==( const IngredientData &ing ) const +{ + return ( ( ing.ingredientID == ingredientID ) && ( ing.groupID == groupID ) ); +} + +Ingredient::Ingredient() : IngredientData() +{} + +Ingredient::Ingredient( const TQString &_name, double _amount, const Unit &_units, int _unitID, int _ingredientID ) : + IngredientData(_name,_amount,_units,_unitID,_ingredientID) +{} + +Ingredient::Ingredient( const IngredientData &d ) : IngredientData(d) +{} + +void Ingredient::setAmount( const TQString &range, bool *ok ) +{ + if ( range.isEmpty() ) { + if ( ok ) *ok = true; + return; + } + + TQStringList ranges = TQStringList::split('-',range); + + TQString amount_min = ranges[0]; + if ( amount_min.isEmpty() ) { + if ( ok ) *ok = false; + return; + } + + TQString amount_max; + switch ( ranges.count() ) { + case 0: + case 1: break; + case 2: amount_max = ranges[1]; + break; + default: + if ( ok ) *ok = false; + return; + } + + MixedNumber nm_min = MixedNumber::fromString( amount_min, ok ); + if ( ok && *ok == false ) return; + + MixedNumber nm_max = MixedNumber::fromString( amount_max, ok ); + if ( ok && *ok == false ) return; + + amount = nm_min.toDouble(); + if ( nm_max > 0 ) + amount_offset = nm_max.toDouble() - amount; +} + +//compare also using the group id because there may be the same ingredient in a list multiple times, but each in a different group +bool Ingredient::operator==( const Ingredient &ing ) const +{ + return ( ( ing.ingredientID == ingredientID ) && ( ing.groupID == groupID ) ); +} + +bool Ingredient::operator<( const Ingredient &ing ) const +{ + return ( TQString::localeAwareCompare( name, ing.name ) < 0 ); +} diff --git a/src/datablocks/ingredient.h b/src/datablocks/ingredient.h new file mode 100644 index 0000000..bdc20d4 --- /dev/null +++ b/src/datablocks/ingredient.h @@ -0,0 +1,63 @@ +/*************************************************************************** +* Copyright (C) 2003 by Unai Garro * +* [email protected] * +* * +* Copyright (C) 2006 Jason Kivlighn <[email protected]> * +* * +* 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. * +***************************************************************************/ +#ifndef INGREDIENT_H +#define INGREDIENT_H + +#include <tqstring.h> +#include <tqvaluelist.h> + +#include "datablocks/unit.h" +#include "datablocks/elementlist.h" + +//###: Is there a better way to get the behavior of a list of Ingredient +// objects as a data member of Ingredient? +class IngredientData +{ +public: + IngredientData(); + IngredientData( const TQString &name, double amount, const Unit &units, int unitID = -1, int ingredientID = -1 ); + + int ingredientID; + TQString name; + double amount; + double amount_offset; + Unit units; + int groupID; + TQString group; + ElementList prepMethodList; + + /** Compare two elements by their id */ + bool operator==( const IngredientData & ) const; +}; + +/** +@author Unai Garro +*/ +class Ingredient : public IngredientData +{ +public: + Ingredient(); + Ingredient( const TQString &name, double amount, const Unit &units, int unitID = -1, int ingredientID = -1 ); + Ingredient( const IngredientData& ); + + TQValueList<IngredientData> substitutes; + + void setAmount( const TQString &range, bool *ok = 0 ); + + /** Compare two elements by their id and groupID */ + bool operator==( const Ingredient & ) const; + + /** This is used for sorting, and so we compare by name */ + bool operator<( const Ingredient & ) const; +}; + +#endif diff --git a/src/datablocks/ingredientlist.cpp b/src/datablocks/ingredientlist.cpp new file mode 100644 index 0000000..7156a48 --- /dev/null +++ b/src/datablocks/ingredientlist.cpp @@ -0,0 +1,272 @@ +/*************************************************************************** +* Copyright (C) 2003 by Unai Garro * +* * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ +#include "ingredientlist.h" + +#include "backends/recipedb.h" + +IngredientList::IngredientList() : TQValueList<Ingredient>() +{} + +IngredientList::~IngredientList() +{} + +bool IngredientList::contains( const Ingredient &ing, bool compareAmount, RecipeDB *database ) const +{ + bool ret = false; + for ( IngredientList::const_iterator it = begin(); it != end(); ++it ) { + if ( (*it).ingredientID == ing.ingredientID ) { + //see if we have enough of the ingredient + //(only if an amount was specified for both) + if ( compareAmount && (*it).amount > 0 && ing.amount > 0 ) { + Ingredient convertedIng; + if ( database->convertIngredientUnits( ing, (*it).units, convertedIng ) ) + ret = (*it).amount >= convertedIng.amount; + else //we couldn't do the conversion + ret = true; + } + else + ret = true; + } + } + if ( !ret ) { + for ( TQValueList<IngredientData>::const_iterator it = ing.substitutes.begin(); it != ing.substitutes.end(); ++it ) { + ret = contains(*it, compareAmount, database); + if ( ret ) break; + } + } + return ret; +} + +bool IngredientList::containsSubSet( IngredientList &il, IngredientList &missing, bool compareAmount, RecipeDB *database ) +{ + missing.empty(); + bool contained = true; + IngredientList::const_iterator it; + + for ( it = il.begin();it != il.end();++it ) { + if ( !contains( *it, compareAmount, database ) ) { + contained = false; + missing.append( *it ); + } + } + + return contained; +} + +bool IngredientList::containsSubSet( IngredientList &il ) const +{ + IngredientList::const_iterator it; + + for ( it = il.begin();it != il.end();++it ) { + if ( !contains( *it ) ) { + return false; + } + } + + return true; +} + +bool IngredientList::containsAny( const IngredientList &list, bool compareAmount, RecipeDB *database ) +{ + for ( IngredientList::const_iterator this_list_it = begin(); this_list_it != end(); ++this_list_it ) { + for ( IngredientList::const_iterator contains_list_it = list.begin(); contains_list_it != list.end(); ++contains_list_it ) { + if ( contains( *contains_list_it, compareAmount, database ) ) + return true; + } + } + + return false; +} + +void IngredientList::empty( void ) +{ + clear(); +} + +int IngredientList::find( int id ) const // Search by id (which uses search by item, with comparison defined on header) +{ + Ingredient i; + i.ingredientID = id; + return findIndex( i ); +} + +Ingredient IngredientList::findByName( const TQString &ing ) const +{ + IngredientList::const_iterator it_end = end(); + for ( IngredientList::const_iterator it = begin(); it != it_end; ++it ) { + if ( ( *it ).name == ing ) + return *it; + + for ( TQValueList<IngredientData>::const_iterator sub_it = ( *it ).substitutes.begin(); sub_it != ( *it ).substitutes.end(); ++sub_it ) { + if ( ( *sub_it ).name == ing ) + return *sub_it; + } + } + + Ingredient el; + el.ingredientID = -1; + return el; +} + +Ingredient IngredientList::findByName( const TQRegExp &rx ) const +{ + IngredientList::const_iterator it_end = end(); + for ( IngredientList::const_iterator it = begin(); it != it_end; ++it ) { + if ( ( *it ).name.find(rx) != -1 ) + return *it; + + for ( TQValueList<IngredientData>::const_iterator sub_it = ( *it ).substitutes.begin(); sub_it != ( *it ).substitutes.end(); ++sub_it ) { + if ( ( *sub_it ).name.find(rx) != -1 ) + return *sub_it; + } + } + + Ingredient el; + el.ingredientID = -1; + return el; +} + +IngredientList::const_iterator IngredientList::find( IngredientList::const_iterator it, int id ) const // Search by id (which uses search by item, with comparison defined on header) +{ + Ingredient i; + i.ingredientID = id; + return TQValueList<Ingredient>::find( it, i ); +} + +IngredientList::iterator IngredientList::find( IngredientList::iterator it, int id ) // Search by id (which uses search by item, with comparison defined on header) +{ + Ingredient i; + i.ingredientID = id; + return TQValueList<Ingredient>::find( it, i ); +} + +IngredientData& IngredientList::findSubstitute( const Ingredient &i ) // Search by id (which uses search by item, with comparison defined on header) +{ + TQValueList<Ingredient>::iterator result = TQValueList<Ingredient>::find(i); + if ( result == end() ) { + for ( IngredientList::iterator it = begin(); it != end(); ++it ) { + TQValueList<IngredientData>::iterator result = (*it).substitutes.find(i); + if ( result != (*it).substitutes.end() ) + return *result; + } + } + return *result; +} + +void IngredientList::removeSubstitute( const Ingredient &i ) +{ + TQValueList<Ingredient>::iterator result = TQValueList<Ingredient>::find(i); + if ( result == end() ) { + for ( IngredientList::iterator it = begin(); it != end(); ++it ) { + TQValueList<IngredientData>::iterator result = (*it).substitutes.find(i); + if ( result != (*it).substitutes.end() ) { + (*it).substitutes.remove(result); + return; + } + } + } + remove(result); +} + +void IngredientList::move( int index1, int index2 ) //moves element in pos index1, to pos after index2 +{ + IngredientList::iterator tmp_it = at( index1 ); + Ingredient tmp_ing( *tmp_it ); + + remove + ( tmp_it ); + + tmp_it = at( index2 ); + insert( tmp_it, tmp_ing ); +} + +void IngredientList::move( int index1, int count1, int index2 ) //moves element in pos index1 and the following count1 items, to pos after index2 +{ + IngredientList::iterator tmp_it = at( index1 ); + IngredientList::iterator after_it = at( index2 + 1 ); + + for ( int i = 0; i < count1; ++i ) + { + Ingredient tmp_ing( *tmp_it ); + + IngredientList::iterator remove_me_it = tmp_it; + ++tmp_it; + remove + ( remove_me_it ); + + insert( after_it, tmp_ing ); + } +} + +IngredientList IngredientList::groupMembers( int id, IngredientList::const_iterator begin ) const +{ + bool first_found = false; + + IngredientList matches; + for ( IngredientList::const_iterator it = begin; it != end(); ++it ) { + if ( ( *it ).groupID == id ) { + matches.append( *it ); + first_found = true; + } + else if ( first_found ) //this is the end of this group... there may be more later though + break; + } + + return matches; +} + +TQValueList<IngredientList::const_iterator> IngredientList::_groupMembers( int id, IngredientList::const_iterator begin ) const +{ + bool first_found = false; + + TQValueList<IngredientList::const_iterator> matches; + for ( IngredientList::const_iterator it = begin; it != end(); ++it ) { + if ( ( *it ).groupID == id ) { + matches << it; + first_found = true; + } + else if ( first_found ) //this is the end of this group... there may be more later though + break; + } + + return matches; +} + +IngredientList IngredientList::firstGroup() +{ + usedGroups.clear(); + + TQValueList<IngredientList::const_iterator> members = _groupMembers( ( *begin() ).groupID, begin() ); + + for ( TQValueList<IngredientList::const_iterator>::const_iterator members_it = members.begin(); members_it != members.end(); ++members_it ) { + usedGroups << *members_it; + } + + return groupMembers( ( *begin() ).groupID, begin() ); +} + +IngredientList IngredientList::nextGroup() +{ + for ( IngredientList::const_iterator it = begin(); it != end(); ++it ) { + if ( usedGroups.find( it ) == usedGroups.end() ) { + TQValueList<IngredientList::const_iterator> members = _groupMembers( ( *it ).groupID, it ); + + for ( TQValueList<IngredientList::const_iterator>::const_iterator members_it = members.begin(); members_it != members.end(); ++members_it ) { + usedGroups << *members_it; + } + + return groupMembers( ( *it ).groupID, it ); + } + } + return IngredientList(); +} diff --git a/src/datablocks/ingredientlist.h b/src/datablocks/ingredientlist.h new file mode 100644 index 0000000..bfd07db --- /dev/null +++ b/src/datablocks/ingredientlist.h @@ -0,0 +1,61 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ +#ifndef INGREDIENTLIST_H +#define INGREDIENTLIST_H + +#include <tqvaluelist.h> +#include <tqregexp.h> + +#include "datablocks/ingredient.h" + +class RecipeDB; + +/** +@author Unai Garro +*/ +class IngredientList: public TQValueList <Ingredient> +{ +public: + IngredientList(); + ~IngredientList(); + bool contains( const Ingredient &ing, bool compareAmount = false, RecipeDB *database = 0 ) const; + bool containsSubSet( IngredientList &il, IngredientList &missing, bool compareAmount = false, RecipeDB *database = 0 ); + bool containsSubSet( IngredientList &il ) const; + bool containsAny( const IngredientList &, bool compareAmount = false, RecipeDB *database = 0 ); + + IngredientList groupMembers( int id, IngredientList::const_iterator begin ) const; + + void move( int index1, int index2 ); + void move( int index1, int count, int index2 ); + void empty( void ); + int find( int id ) const; + Ingredient findByName( const TQString & ) const; + Ingredient findByName( const TQRegExp & ) const; + IngredientList::const_iterator find( IngredientList::const_iterator, int id ) const; + IngredientList::iterator find( IngredientList::iterator, int id ); + + /** Warning, returns an invalid reference if no ingredient is found. Must check prior + * to calling this function if the ingredient exists. + */ + IngredientData& findSubstitute( const Ingredient & ); + void removeSubstitute( const Ingredient & ); + + IngredientList firstGroup(); + IngredientList nextGroup(); + +private: + TQValueList<IngredientList::const_iterator> _groupMembers( int id, IngredientList::const_iterator begin ) const; + TQValueList<IngredientList::const_iterator> usedGroups; +}; + +#endif diff --git a/src/datablocks/ingredientproperty.cpp b/src/datablocks/ingredientproperty.cpp new file mode 100644 index 0000000..d8c621e --- /dev/null +++ b/src/datablocks/ingredientproperty.cpp @@ -0,0 +1,32 @@ +/*************************************************************************** +* Copyright (C) 2003 by Unai Garro * +* [email protected] * +* * +* 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. * +***************************************************************************/ +#include "datablocks/ingredientproperty.h" + +IngredientProperty::IngredientProperty() +{ + id = -1; + ingredientID = -1; + amount = 0.0; +} + +IngredientProperty::IngredientProperty( const TQString &_name, const TQString &_units, int _id ) : + id( _id ), + name( _name ), + units( _units ) +{} + +IngredientProperty::~IngredientProperty() +{} + +bool IngredientProperty::operator==( const IngredientProperty &prop ) const +{ + return ( prop.id == id ); +} + diff --git a/src/datablocks/ingredientproperty.h b/src/datablocks/ingredientproperty.h new file mode 100644 index 0000000..5b5fd55 --- /dev/null +++ b/src/datablocks/ingredientproperty.h @@ -0,0 +1,37 @@ +/*************************************************************************** +* Copyright (C) 2003 by Unai Garro * +* [email protected] * +* * +* 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. * +***************************************************************************/ +#ifndef INGREDIENTPROPERTY_H +#define INGREDIENTPROPERTY_H + +#include <tqstring.h> + +#include "unit.h" + +/** +@author Unai Garro +*/ +class IngredientProperty +{ +public: + IngredientProperty(); + IngredientProperty( const TQString &name, const TQString &units, int id = -1 ); + ~IngredientProperty(); + int id; // The property's id + int ingredientID; // (Optional) reference to the ingredient to which is attached + TQString name; // Name of the property + TQString units; // The units that the property uses + Unit perUnit; // stores the unit ID, name, and type of the per units. + double amount; // Stores the amount, in the case of being attached to an ingredient. If not attached to any, you can set it to -1 preferably. That's the case in which the property is treated as a characteristic any without value (amount). + + /** Compare two elements by their id */ + bool operator==( const IngredientProperty & ) const; +}; + +#endif diff --git a/src/datablocks/ingredientpropertylist.cpp b/src/datablocks/ingredientpropertylist.cpp new file mode 100644 index 0000000..40d63be --- /dev/null +++ b/src/datablocks/ingredientpropertylist.cpp @@ -0,0 +1,53 @@ +/*************************************************************************** +* Copyright (C) 2003 by Unai Garro * +* [email protected] * +* * +* 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. * +***************************************************************************/ + +#include "datablocks/ingredientpropertylist.h" + +IngredientPropertyList::IngredientPropertyList() +{} + + +IngredientPropertyList::~IngredientPropertyList() +{} + +IngredientPropertyList::const_iterator IngredientPropertyList::find( int id ) +{ + IngredientProperty ip; + ip.id = id; + return TQValueList<IngredientProperty>::find( ip ); +} + +int IngredientPropertyList::findByName( const TQString &name ) +{ + IngredientPropertyList::const_iterator prop_it; + for ( prop_it = begin(); prop_it != end(); ++prop_it ) { + if ( (*prop_it).name == name ) + return (*prop_it).id; + } + + return -1; +} + +void IngredientPropertyList::divide( double units_of_yield_type ) +{ + IngredientPropertyList::iterator prop_it; + for ( prop_it = begin(); prop_it != end(); ++prop_it ) + (*prop_it).amount /= units_of_yield_type; +} + +void IngredientPropertyList::filter( int ingredientID, IngredientPropertyList *filteredList ) +{ + filteredList->clear(); + IngredientPropertyList::const_iterator prop_it; + for ( prop_it = begin(); prop_it != end(); ++prop_it ) { + if ( (*prop_it).ingredientID == ingredientID ) + filteredList->append( *prop_it ); + } +} diff --git a/src/datablocks/ingredientpropertylist.h b/src/datablocks/ingredientpropertylist.h new file mode 100644 index 0000000..5a8811d --- /dev/null +++ b/src/datablocks/ingredientpropertylist.h @@ -0,0 +1,30 @@ +/*************************************************************************** +* Copyright (C) 2003 by Unai Garro * +* [email protected] * +* * +* 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. * +***************************************************************************/ +#ifndef INGREDIENTPROPERTYLIST_H +#define INGREDIENTPROPERTYLIST_H + +#include <tqvaluelist.h> + +#include "datablocks/ingredientproperty.h" + +class IngredientPropertyList : public TQValueList<IngredientProperty> +{ +public: + IngredientPropertyList(); + + ~IngredientPropertyList(); + + void divide( double units_of_yield_type ); + IngredientPropertyList::const_iterator find( int id ); + int findByName( const TQString & ); + void filter( int ingredientID, IngredientPropertyList *filteredList ); +}; + +#endif diff --git a/src/datablocks/kreborder.cpp b/src/datablocks/kreborder.cpp new file mode 100644 index 0000000..4109f1e --- /dev/null +++ b/src/datablocks/kreborder.cpp @@ -0,0 +1,17 @@ +/*************************************************************************** +* Copyright (C) 2004 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "kreborder.h" + +//typedef enum KreBorderStyle { None = 0, Dotted, Dashed, Solid, Double, Groove, Ridge, Inset, Outset }; + +KreBorder::KreBorder( int w, const TQString & s, const TQColor &c ) : width( w ), style( s ), color( c ) +{ +} diff --git a/src/datablocks/kreborder.h b/src/datablocks/kreborder.h new file mode 100644 index 0000000..0e535b6 --- /dev/null +++ b/src/datablocks/kreborder.h @@ -0,0 +1,29 @@ +/*************************************************************************** +* Copyright (C) 2004 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef KREBORDER_H +#define KREBORDER_H + +#include <tqcolor.h> +#include <tqstring.h> + +//typedef enum KreBorderStyle { None = 0, Dotted, Dashed, Solid, Double, Groove, Ridge, Inset, Outset }; + +class KreBorder +{ +public: + KreBorder( int w = 1, const TQString & s = "none", const TQColor &c = TQt::black ); + + int width; + TQString style; + TQColor color; +}; + +#endif //KREBORDER_H diff --git a/src/datablocks/mixednumber.cpp b/src/datablocks/mixednumber.cpp new file mode 100644 index 0000000..17a9047 --- /dev/null +++ b/src/datablocks/mixednumber.cpp @@ -0,0 +1,289 @@ +/*************************************************************************** +* Copyright (C) 2003 by krecipes.sourceforge.net authors * +* * +* * +* 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. * +***************************************************************************/ +#include "mixednumber.h" + +#include <tqregexp.h> + +#include <tdeglobal.h> +#include <tdelocale.h> +#include <kdebug.h> + +TQString beautify( const TQString &num ) +{ + TQString copy( num ); + copy.remove( TQRegExp( TQString( "(%1){0,1}0+$" ).arg( TQRegExp::escape( TDEGlobal::locale() ->decimalSymbol() ) ) ) ); + return copy; +} + +MixedNumber::MixedNumber() : + m_whole( 0 ), + m_numerator( 0 ), + m_denominator( 1 ), + locale( TDEGlobal::locale() ) +{} + +MixedNumber::MixedNumber( int whole, int numerator, int denominator ) : + m_whole( whole ), + m_numerator( numerator ), + m_denominator( denominator ), + locale( TDEGlobal::locale() ) +{} + +MixedNumber::MixedNumber( double decimal, double precision ) : + locale( TDEGlobal::locale() ) +{ + // find nearest fraction + int intPart = static_cast<int>( decimal ); + decimal -= static_cast<double>( intPart ); + + MixedNumber low( 0, 0, 1 ); // "A" = 0/1 + MixedNumber high( 0, 1, 1 ); // "B" = 1/1 + + for ( int i = 0; i < 100; ++i ) { + double testLow = low.denominator() * decimal - low.numerator(); + double testHigh = high.numerator() - high.denominator() * decimal; + + if ( testHigh < precision * high.denominator() ) + break; // high is answer + if ( testLow < precision * low.denominator() ) { // low is answer + high = low; + break; + } + + if ( i & 1 ) { // odd step: add multiple of low to high + double test = testHigh / testLow; + int count = ( int ) test; // "N" + int num = ( count + 1 ) * low.numerator() + high.numerator(); + int denom = ( count + 1 ) * low.denominator() + high.denominator(); + + if ( ( num > 0x8000 ) || ( denom > 0x10000 ) ) + break; + + high.setNumerator( num - low.numerator() ); // new "A" + high.setDenominator( denom - low.denominator() ); + low.setNumerator( num ); // new "B" + low.setDenominator( denom ); + } + else { // even step: add multiple of high to low + double test = testLow / testHigh; + int count = ( int ) test; // "N" + int num = low.numerator() + ( count + 1 ) * high.numerator(); + int denom = low.denominator() + ( count + 1 ) * high.denominator(); + + if ( ( num > 0x10000 ) || ( denom > 0x10000 ) ) + break; + + low.setNumerator( num - high.numerator() ); // new "A" + low.setDenominator( denom - high.denominator() ); + high.setNumerator( num ); // new "B" + high.setDenominator( denom ); + } + } + + m_numerator = high.numerator(); + m_denominator = high.denominator(); + m_whole = intPart; +} + +MixedNumber::~MixedNumber() +{} + +int MixedNumber::getNumerator( const TQString &input, int space_index, int slash_index, bool *ok ) +{ + return input.mid( space_index + 1, slash_index - space_index - 1 ).toInt( ok ); +} + +int MixedNumber::getDenominator( const TQString &input, int slash_index, bool *ok ) +{ + return input.mid( slash_index + 1, input.length() ).toInt( ok ); +} + +MixedNumber MixedNumber::fromString( const TQString &str, bool *ok, bool locale_aware ) +{ + TQString input = str.stripWhiteSpace(); + if ( input.isEmpty() ) { + if ( ok ) { + *ok = true; + } + return MixedNumber(); + } + + TDELocale *locale = TDEGlobal::locale(); + + bool num_ok; + + int whole; + int numerator; + int denominator; + + int space_index = input.find( " " ); + int slash_index = input.find( "/" ); + + if ( space_index == -1 ) { + if ( slash_index == -1 ) //input contains no fractional part + { + TQString decimal_symbol = ( locale_aware ) ? locale->decimalSymbol() : "."; + if ( input.endsWith( decimal_symbol ) ) + { + if ( ok ) { + *ok = false; + } + return MixedNumber(); + } + + double decimal = ( locale_aware ) ? locale->readNumber( input, &num_ok ) : input.toDouble( &num_ok ); + + if ( !num_ok ) + { + if ( ok ) { + *ok = false; + } + return MixedNumber(); + } + + if ( ok ) + { + *ok = true; + } + return MixedNumber( decimal ); + } + else //input just contains a fraction + { + whole = 0; + + numerator = MixedNumber::getNumerator( input, space_index, slash_index, &num_ok ); + if ( !num_ok ) { + if ( ok ) { + *ok = false; + } + return MixedNumber(); + } + + denominator = MixedNumber::getDenominator( input, slash_index, &num_ok ); + if ( !num_ok || denominator == 0 ) { + if ( ok ) { + *ok = false; + } + return MixedNumber(); + } + + if ( ok ) { + *ok = true; + } + return MixedNumber( whole, numerator, denominator ); + } + if ( ok ) { + *ok = false; + } + return MixedNumber(); + } + + whole = input.mid( 0, space_index ).toInt( &num_ok ); + if ( !num_ok ) { + if ( ok ) { + *ok = false; + } + return MixedNumber(); + } + + numerator = MixedNumber::getNumerator( input, space_index, slash_index, &num_ok ); + if ( !num_ok ) { + if ( ok ) { + *ok = false; + } + return MixedNumber(); + } + + denominator = MixedNumber::getDenominator( input, slash_index, &num_ok ); + if ( !num_ok || denominator == 0 ) { + if ( ok ) { + *ok = false; + } + return MixedNumber(); + } + + if ( ok ) { + *ok = true; + } + return MixedNumber( whole, numerator, denominator ); +} + +TQString MixedNumber::toString( Format format, bool locale_aware ) const +{ + if ( format == DecimalFormat ) { + if ( locale_aware ) + return beautify( locale->formatNumber( toDouble(), 5 ) ); + else + return TQString::number( toDouble() ); + } + + if ( m_numerator == 0 && m_whole == 0 ) + return TQString( "0" ); + + + TQString result; + + if ( m_whole != 0 ) { + result += TQString::number( m_whole ); + if ( m_numerator != 0 ) + result += " "; + } + + if ( m_numerator != 0 ) + result += TQString::number( m_numerator ) + "/" + TQString::number( m_denominator ); + + return result; +} + +bool MixedNumber::operator!=( const MixedNumber &fraction ) +{ + return ( fraction.toDouble() != toDouble() ); +} + +MixedNumber& MixedNumber::operator+=( const MixedNumber &fraction ) +{ + m_numerator = ( m_numerator * fraction.m_denominator ) + ( m_denominator * fraction.m_numerator ); + m_denominator = m_denominator * fraction.m_denominator; + m_whole += fraction.m_whole; + simplify(); + + return *this; +} + +MixedNumber& MixedNumber::operator+=( double d ) +{ + MixedNumber mn(d); + *this += mn; + return *this; +} + +void MixedNumber::simplify() +{ + int divisor = gcd( m_numerator, m_denominator ); + m_numerator /= divisor; + m_denominator /= divisor; +} + +double MixedNumber::toDouble() const +{ + return static_cast<double>( m_whole ) + ( static_cast<double>( m_numerator ) / static_cast<double>( m_denominator ) ); +} + +int MixedNumber::gcd( int n, int m ) +{ + int r; + while ( n != 0 ) { + r = m % n; + m = n; + n = r; + } + + return m; +} diff --git a/src/datablocks/mixednumber.h b/src/datablocks/mixednumber.h new file mode 100644 index 0000000..34a0f8b --- /dev/null +++ b/src/datablocks/mixednumber.h @@ -0,0 +1,126 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef MIXEDNUMBER_H +#define MIXEDNUMBER_H + +#include <tqstring.h> + +/** remove any extra zeros on the end of the string and the decimal if a whole number */ +TQString beautify( const TQString &num ); + +class TDELocale; + +/** A class to hold and manipulate a mixed number. + * @author Jason Kivlighn + */ +class MixedNumber +{ +public: + MixedNumber( int whole, int numerator, int denominator ); + + /** Create a mixed number from the given @param decimal. This uses a method that will + * approximate the actual fraction with precision equal to @param precision. + * The closer @param precision is to zero without being zero, the more precisely + * it will try to approximate the fraction. + */ + MixedNumber( double decimal, double precision = 1e-6 ); + + /** Creates a mixed number with an initial value of zero. */ + MixedNumber(); + + ~MixedNumber(); + + MixedNumber& operator+=( const MixedNumber & ); + MixedNumber& operator+=( double ); + bool operator!=( const MixedNumber &fraction ); + bool operator>( double d ) + { + return ( toDouble() > d ); + } + + enum Format { DecimalFormat, MixedNumberFormat }; + + /** The input as a decimal. */ + double toDouble() const; + + /** Returns the fraction as a string */ + TQString toString( Format = MixedNumberFormat, bool locale_aware = true ) const; + + /** The whole part of the input */ + int whole() const + { + return m_whole; + } + + /** The numerator of the fractional part of the input. */ + int numerator() const + { + return m_numerator; + } + + /** The denominator of the fractional part of the input. */ + int denominator() const + { + return m_denominator; + } + + void setNumerator( int n ) + { + m_numerator = n; + } + void setDenominator( int d ) + { + m_denominator = d; + } + + /** Ensure that the fraction is simplified to its lowest terms. */ + void simplify(); + + /** Parses the given TQString as a mixed number. The input can be + * expressed as a mixed number in the form "a b/c", or as a decimal. + */ + static MixedNumber fromString( const TQString &input, bool *ok = 0, bool locale_aware = true ); + +private: + static int getNumerator( const TQString &input, int space_index, int slash_index, bool *ok ); + static int getDenominator( const TQString &input, int slash_index, bool *ok ); + + int gcd( int, int ); + + int m_whole; + int m_numerator; + int m_denominator; + + TDELocale *locale; +}; + +inline const MixedNumber operator+( const MixedNumber &mn1, const MixedNumber &mn2 ) +{ + MixedNumber tmp = mn1; + tmp += mn2; + return tmp; +} + + +inline const MixedNumber operator+( double d, const MixedNumber &mn ) +{ + MixedNumber tmp = mn; + tmp += d; + return tmp; +} + +inline const MixedNumber operator+( const MixedNumber &mn, double d ) +{ + return operator+(d,mn); +} + + +#endif //MIXEDNUMBER_H diff --git a/src/datablocks/rating.cpp b/src/datablocks/rating.cpp new file mode 100644 index 0000000..83f0d9b --- /dev/null +++ b/src/datablocks/rating.cpp @@ -0,0 +1,86 @@ +/*************************************************************************** +* Copyright (C) 2005 by Jason Kivlighn * +* [email protected] * +* * +* 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. * +***************************************************************************/ + +#include "rating.h" + +#include <tqpainter.h> +#include <tqbitmap.h> + +#include <kiconloader.h> + +TQPixmap Rating::starsPixmap( double stars_d, bool include_empty ) +{ + int stars = tqRound(stars_d * 2); //multiply by two to make it easier to work with half-stars + + TQPixmap star = UserIcon(TQString::fromLatin1("star_on")); + TQPixmap star_off; + if ( include_empty ) + star_off = UserIcon(TQString::fromLatin1("star_off")); + + int pixmapWidth; + if ( include_empty ) + pixmapWidth = 18*5; + else + pixmapWidth = 18*(stars/2)+((stars%2==1)?9:0); + + TQPixmap generatedPixmap(pixmapWidth,18); + + if ( !generatedPixmap.isNull() ) { //there aren't zero stars + generatedPixmap.fill(); + TQPainter painter( &generatedPixmap ); + + int pixmapWidth = 18*(stars/2)+((stars%2==1)?9:0); + if ( include_empty ) + painter.drawTiledPixmap(0,0,18*5,18,star_off); //fill with empty stars + painter.drawTiledPixmap(0,0,pixmapWidth,18,star); //write over the empty stars to show the rating + } + + generatedPixmap.setMask( generatedPixmap.createHeuristicMask() ); + + return generatedPixmap; +} + +void Rating::append( const RatingCriteria &rc ) +{ + ratingCriteriaList.append( rc ); +} + +double Rating::average() const +{ + double sum = 0; + int count = 0; + for ( RatingCriteriaList::const_iterator rc_it = ratingCriteriaList.begin(); rc_it != ratingCriteriaList.end(); ++rc_it ) { + count++; + sum += (*rc_it).stars; + } + + if ( count > 0 ) + return sum/count; + else + return -1; +} + + +double RatingList::average() +{ + int rating_total = 0; + double rating_sum = 0; + for ( RatingList::const_iterator rating_it = begin(); rating_it != end(); ++rating_it ) { + for ( RatingCriteriaList::const_iterator rc_it = (*rating_it).ratingCriteriaList.begin(); rc_it != (*rating_it).ratingCriteriaList.end(); ++rc_it ) { + rating_total++; + rating_sum += (*rc_it).stars; + } + } + + if ( rating_total > 0 ) + return rating_sum/rating_total; + else + return -1; +} diff --git a/src/datablocks/rating.h b/src/datablocks/rating.h new file mode 100644 index 0000000..60e7693 --- /dev/null +++ b/src/datablocks/rating.h @@ -0,0 +1,55 @@ +/*************************************************************************** +* Copyright (C) 2005 by Jason Kivlighn * +* [email protected] * +* * +* 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. * +***************************************************************************/ + +#ifndef RATING_H +#define RATING_H + +#include <tqvaluelist.h> +#include <tqstring.h> + +#include <tqpixmap.h> + +class RatingCriteria +{ +public: + RatingCriteria() : id(-1), stars(0.0){} + + int id; + TQString name; + double stars; +}; + +typedef TQValueList< RatingCriteria > RatingCriteriaList; + +class Rating +{ +public: + Rating() : id(-1){} + + static TQPixmap starsPixmap( double stars_d, bool include_empty = false ); + + void append( const RatingCriteria & ); + + double average() const; + + int id; + TQString comment; + TQString rater; + + RatingCriteriaList ratingCriteriaList; +}; + +class RatingList : public TQValueList< Rating > +{ +public: + double average(); +}; + +#endif diff --git a/src/datablocks/recipe.cpp b/src/datablocks/recipe.cpp new file mode 100644 index 0000000..dd00f07 --- /dev/null +++ b/src/datablocks/recipe.cpp @@ -0,0 +1,53 @@ +/*************************************************************************** +* Copyright (C) 2003-2004 by * +* Unai Garro ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ +#include "datablocks/recipe.h" + +Recipe::Recipe() +{ + empty(); //Create & initialize the recipe empty originally +} + +Recipe::~Recipe() +{} + +void Recipe::empty( void ) +{ + recipeID = -1; + + yield.amount = 1; + yield.amount_offset = 0; + yield.type = TQString::null; + + title = TQString::null; + instructions = TQString::null; + photo = TQPixmap(); + ingList.empty(); + categoryList.clear(); + authorList.clear(); + ratingList.clear(); + prepTime = TQTime( 0, 0 ); +} + + +TQString Yield::amountToString() const +{ + TQString ret = TQString::number(amount); + if ( amount_offset > 0 ) + ret += "-"+TQString::number(amount+amount_offset); + + return ret; +} + +TQString Yield::toString() const +{ + return amountToString() + " " + type; +} + diff --git a/src/datablocks/recipe.h b/src/datablocks/recipe.h new file mode 100644 index 0000000..c1c8aab --- /dev/null +++ b/src/datablocks/recipe.h @@ -0,0 +1,71 @@ +/*************************************************************************** +* Copyright (C) 2003-2004 by * +* Unai Garro ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ +#ifndef RECIPE_H +#define RECIPE_H + +#include <tqstring.h> +#include <tqpixmap.h> +#include <tqdatetime.h> + +#include "ingredientlist.h" +#include "datablocks/rating.h" +#include "datablocks/elementlist.h" +#include "datablocks/ingredientpropertylist.h" + +class Yield +{ +public: + Yield() : amount(1), amount_offset(0), type(TQString::null), type_id(-1){} + + TQString amountToString() const; + TQString toString() const; + + double amount; + double amount_offset; + TQString type; + int type_id; +}; + +/** +@author Unai Garro +*/ +class Recipe +{ +public: + Recipe(); + ~Recipe(); + // Public variables + + int recipeID; + Yield yield; + TQString title; + TQString instructions; + TQPixmap photo; + IngredientList ingList; + ElementList categoryList; // id+name + ElementList authorList; //authors' id+name + TQTime prepTime; + + TQDateTime ctime; + TQDateTime mtime; + TQDateTime atime; + + RatingList ratingList; + IngredientPropertyList properties; + + // Public methods + void empty( void ); + + + +}; + +#endif diff --git a/src/datablocks/recipelist.cpp b/src/datablocks/recipelist.cpp new file mode 100644 index 0000000..5a7d583 --- /dev/null +++ b/src/datablocks/recipelist.cpp @@ -0,0 +1,18 @@ +/*************************************************************************** +* Copyright (C) 2003 by Unai Garro ([email protected]) * +* * +* * +* 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. * +***************************************************************************/ + +#include "recipelist.h" + +RecipeList::RecipeList() : TQValueList <Recipe>() +{} + +RecipeList::~RecipeList() +{} + diff --git a/src/datablocks/recipelist.h b/src/datablocks/recipelist.h new file mode 100644 index 0000000..ac8cc99 --- /dev/null +++ b/src/datablocks/recipelist.h @@ -0,0 +1,28 @@ +/*************************************************************************** +* Copyright (C) 2003 by Unai Garro [email protected] * +* * +* * +* 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. * +***************************************************************************/ + +#ifndef RECIPELIST_H +#define RECIPELIST_H + +#include "datablocks/recipe.h" + +#include <tqvaluelist.h> + +/** +@author Unai Garro +*/ +class RecipeList: public TQValueList <Recipe> +{ +public: + RecipeList(); + ~RecipeList(); +}; + +#endif diff --git a/src/datablocks/unit.cpp b/src/datablocks/unit.cpp new file mode 100644 index 0000000..ab670f3 --- /dev/null +++ b/src/datablocks/unit.cpp @@ -0,0 +1,75 @@ +/*************************************************************************** +* Copyright (C) 2004 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "unit.h" + +Unit::Unit() : type(Unit::Other), id( -1 ) +{} + +Unit::Unit( const TQString &_name, const TQString &_plural, int _id ) : + type(Unit::Other), + id( _id ), + name( _name ), + plural( _plural ) +{} + +Unit::Unit( const TQString &_name, double amount ) : type(Unit::Other) +{ + if ( amount > 1 ) + plural = _name; + else + name = _name; +} + +TQString Unit::determineName( double amount, bool useAbbrev ) const +{ + if ( useAbbrev ) { + TQString unit = ( amount > 1 ) ? plural_abbrev : name_abbrev; + if ( unit.isEmpty() ) + unit = ( amount > 1 ) ? plural : name; + return unit; + } + else + return ( amount > 1 ) ? plural : name; +} + +bool Unit::operator==( const Unit &u ) const +{ + //treat TQString::null and "" as the same + TQString plural_test1 = u.plural.lower(); + if ( plural_test1.isNull() ) + plural_test1 = ""; + + TQString plural_test2 = plural.lower(); + if ( plural_test2.isNull() ) + plural_test2 = ""; + + TQString single_test1 = u.name.lower(); + if ( single_test1.isNull() ) + single_test1 = ""; + + TQString single_test2 = name.lower(); + if ( single_test2.isNull() ) + single_test2 = ""; + + if ( plural_test1.isEmpty() && plural_test2.isEmpty() && single_test1.isEmpty() && single_test2.isEmpty() ) + return true; + else if ( plural_test1.isEmpty() && plural_test2.isEmpty() ) + return single_test1 == single_test2; + else if ( single_test1.isEmpty() && single_test2.isEmpty() ) + return plural_test1 == plural_test2; + else + return ( plural_test1 == plural_test2 || single_test1 == single_test2 ); +} + +bool Unit::operator<( const Unit &u ) const +{ + return ( TQString::localeAwareCompare( name.lower(), u.name.lower() ) < 0 ); +} diff --git a/src/datablocks/unit.h b/src/datablocks/unit.h new file mode 100644 index 0000000..f948cf8 --- /dev/null +++ b/src/datablocks/unit.h @@ -0,0 +1,45 @@ +/*************************************************************************** +* Copyright (C) 2004-2006 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef UNIT_H +#define UNIT_H + +#include <tqstring.h> +#include <tqvaluelist.h> + +class Unit +{ +public: + Unit(); + Unit( const TQString &name, const TQString &plural, int id = -1 ); + + typedef enum { All = -1, Other = 0, Mass, Volume } Type; + + /** Use @param amount to determine whether to use @param name as the plural or singlular form */ + Unit( const TQString &name, double amount ); + + bool operator==( const Unit &u ) const; + bool operator<( const Unit &u ) const; + + TQString determineName( double amount, bool useAbbrev ) const; + + Type type; + + int id; + TQString name; + TQString plural; + + TQString name_abbrev; + TQString plural_abbrev; +}; + +typedef TQValueList< Unit > UnitList; + +#endif //UNIT_H diff --git a/src/datablocks/unitratio.cpp b/src/datablocks/unitratio.cpp new file mode 100644 index 0000000..fb531cd --- /dev/null +++ b/src/datablocks/unitratio.cpp @@ -0,0 +1,30 @@ +/*************************************************************************** +* Copyright (C) 2003 by Unai Garro * +* [email protected] * +* * +* 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. * +***************************************************************************/ +#include "unitratio.h" +#include "tdelocale.h" + +UnitRatio::UnitRatio() +{ + uID1 = -1; + uID2 = -1; + ratio = -1; +} + +UnitRatio::UnitRatio( const UnitRatio &ur ) +{ + uID1 = ur.uID1; + uID2 = ur.uID2; + ratio = ur.ratio; +} + +UnitRatio::~UnitRatio() +{} + + diff --git a/src/datablocks/unitratio.h b/src/datablocks/unitratio.h new file mode 100644 index 0000000..dd327f1 --- /dev/null +++ b/src/datablocks/unitratio.h @@ -0,0 +1,37 @@ +/*************************************************************************** +* Copyright (C) 2003 by Unai Garro * +* [email protected] * +* * +* 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. * +***************************************************************************/ +#ifndef UNITRATIO_H +#define UNITRATIO_H + +/** +@author Unai Garro +*/ +class UnitRatio +{ +public: + + UnitRatio(); + UnitRatio( const UnitRatio &ur ); + ~UnitRatio(); + + bool operator!=( const UnitRatio &r ) const + { + return !( r.uID1 == uID1 && r.uID2 == uID2 ); + } + bool operator==( const UnitRatio &r ) const + { + return ( r.uID1 == uID1 && r.uID2 == uID2 ); + } + + int uID1, uID2; + double ratio; +}; + +#endif diff --git a/src/datablocks/unitratiolist.cpp b/src/datablocks/unitratiolist.cpp new file mode 100644 index 0000000..6955005 --- /dev/null +++ b/src/datablocks/unitratiolist.cpp @@ -0,0 +1,34 @@ +/*************************************************************************** +* Copyright (C) 2003 by Unai Garro * +* [email protected] * +* * +* 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. * +***************************************************************************/ +#include "unitratiolist.h" +#include "tdelocale.h" + + +UnitRatioList::UnitRatioList() +{} + + +UnitRatioList::~UnitRatioList() +{} + +double UnitRatioList::getRatio( int uid1, int uid2 ) +{ + if ( uid1 == uid2 ) + return ( 1.0 ); + else { + for ( UnitRatioList::const_iterator ur_it = begin();ur_it != end(); ++ur_it ) { + if ( ( *ur_it ).uID1 == uid1 && ( *ur_it ).uID2 == uid2 ) + return ( ( *ur_it ).ratio ); + else if ( ( *ur_it ).uID1 == uid2 && ( *ur_it ).uID2 == uid1 ) + return ( 1.0 / ( *ur_it ).ratio ); + } + return ( -1.0 ); + } +} diff --git a/src/datablocks/unitratiolist.h b/src/datablocks/unitratiolist.h new file mode 100644 index 0000000..9b74e0a --- /dev/null +++ b/src/datablocks/unitratiolist.h @@ -0,0 +1,33 @@ +/*************************************************************************** +* Copyright (C) 2003 by Unai Garro * +* [email protected] * +* * +* 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. * +***************************************************************************/ +#ifndef UNITRATIOLIST_H +#define UNITRATIOLIST_H +#include <tqvaluelist.h> +#include "unitratio.h" + + +/** +@author Unai Garro +*/ +class UnitRatioList : public TQValueList <UnitRatio> +{ +public: + UnitRatioList(); + ~UnitRatioList(); + + void add + ( const UnitRatio &r ) + { + append( r ); + } + double getRatio( int uid1, int uid2 ); +}; + +#endif diff --git a/src/datablocks/weight.cpp b/src/datablocks/weight.cpp new file mode 100644 index 0000000..08ba695 --- /dev/null +++ b/src/datablocks/weight.cpp @@ -0,0 +1,14 @@ +/*************************************************************************** +* Copyright (C) 2006 Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "weight.h" + +Weight::Weight() : id(-1), prepMethodID(-1) +{} + diff --git a/src/datablocks/weight.h b/src/datablocks/weight.h new file mode 100644 index 0000000..c5042bd --- /dev/null +++ b/src/datablocks/weight.h @@ -0,0 +1,41 @@ +/*************************************************************************** +* Copyright (C) 2006 Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef WEIGHT_H +#define WEIGHT_H + +#include <tqstring.h> +#include <tqvaluelist.h> + +#include "datablocks/elementlist.h" + +class Weight +{ +public: + Weight(); + + int id; + int ingredientID; + int perAmountUnitID; + TQString perAmountUnit; + double perAmount; + int weightUnitID; + double weight; + TQString weightUnit; + int prepMethodID; + TQString prepMethod; +}; + +class WeightList : public TQValueList<Weight> +{ +public: + WeightList() : TQValueList<Weight>(){} +}; + +#endif diff --git a/src/dialogs/Makefile.am b/src/dialogs/Makefile.am new file mode 100644 index 0000000..23f80e1 --- /dev/null +++ b/src/dialogs/Makefile.am @@ -0,0 +1,38 @@ +## Makefile.am for krecipes + +# this is the program that gets installed. it's name is used for all +# of the other Makefile.am variables + +# set the include path for X, tqt and TDE +INCLUDES = -I$(srcdir)/.. $(all_includes) + +KDE_CXXFLAGS = $(USE_EXCEPTIONS) + +noinst_LTLIBRARIES=libkrecipesdialogs.la +libkrecipesdialogs_la_SOURCES= \ + advancedsearchdialog.cpp recipeimportdialog.cpp \ + dietwizarddialog.cpp recipeinputdialog.cpp \ + recipeviewdialog.cpp selectrecipedialog.cpp \ + ingredientsdialog.cpp selectunitdialog.cpp \ + createelementdialog.cpp propertiesdialog.cpp \ + createpropertydialog.cpp selectpropertydialog.cpp \ + unitsdialog.cpp dependanciesdialog.cpp \ + shoppinglistdialog.cpp shoppinglistviewdialog.cpp \ + selectcategoriesdialog.cpp categorieseditordialog.cpp \ + authorsdialog.cpp selectauthorsdialog.cpp \ + resizerecipedialog.cpp \ + dietviewdialog.cpp ingredientmatcherdialog.cpp \ + usdadatadialog.cpp prepmethodsdialog.cpp \ + createcategorydialog.cpp borderdialog.cpp \ + refineshoppinglistdialog.cpp pagesetupdialog.cpp \ + dbimportdialog.cpp createunitdialog.cpp \ + setupdisplay.cpp \ + ingredientparserdialog.cpp ingredientgroupsdialog.cpp \ + editratingdialog.cpp similarcategoriesdialog.cpp \ + conversiondialog.cpp createingredientweightdialog.cpp \ + recipeprintpreview.cpp + +libkrecipesdialogs_la_METASOURCES=AUTO + +#the library search path. +libkrecipesdialogs_la_LDFLAGS = $(KDE_RPATH) $(all_libraries) diff --git a/src/dialogs/advancedsearchdialog.cpp b/src/dialogs/advancedsearchdialog.cpp new file mode 100644 index 0000000..9f4864d --- /dev/null +++ b/src/dialogs/advancedsearchdialog.cpp @@ -0,0 +1,932 @@ +/*************************************************************************** +* Copyright (C) 2003-2004 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "advancedsearchdialog.h" + +#include <tqvariant.h> +#include <tqpushbutton.h> +#include <tqtabwidget.h> +#include <tqwidget.h> +#include <tqcheckbox.h> +#include <tqframe.h> +#include <tqcombobox.h> +#include <tqheader.h> +#include <tqlistview.h> +#include <tqgroupbox.h> +#include <tqspinbox.h> +#include <tqdatetimeedit.h> +#include <kpushbutton.h> +#include <tqlayout.h> +#include <tqtooltip.h> +#include <tqwhatsthis.h> +#include <tqlabel.h> +#include <tqlineedit.h> +#include <tqscrollview.h> +#include <tqhbox.h> +#include <tqradiobutton.h> +#include <tqbuttongroup.h> + +#include <tdeapplication.h> +#include <kcursor.h> +#include <kdebug.h> +#include <kiconloader.h> +#include <tdelistview.h> +#include <tdelocale.h> +#include <tdeconfig.h> +#include <tdeglobal.h> +#include <tdepopupmenu.h> + +#include "backends/recipedb.h" +#include "backends/searchparameters.h" +#include "recipeactionshandler.h" +#include "widgets/recipelistview.h" +#include "widgets/kdateedit.h" +#include "widgets/ratingwidget.h" +#include "widgets/fractioninput.h" +#include "widgets/criteriacombobox.h" + +#include "profiling.h" + +AdvancedSearchDialog::AdvancedSearchDialog( TQWidget *parent, RecipeDB *db ) : TQWidget( parent ), + database( db ) +{ + /// + ///BEGIN OF AUTOMATICALLY GENERATED GUI CODE/// + /// + AdvancedSearchDialogLayout = new TQHBoxLayout( this, 5, 3, "AdvancedSearchDialogLayout"); + + layout7 = new TQVBoxLayout( 0, 0, 3, "layout7"); + + textLabel1_4 = new TQLabel( this, "textLabel1_4" ); + layout7->addWidget( textLabel1_4 ); + + scrollView1 = new TQScrollView( this, "scrollView1" ); + scrollView1->enableClipper(true); + + parametersFrame = new TQFrame( scrollView1, "parametersFrame" ); + parametersFrame->setSizePolicy( TQSizePolicy( (TQSizePolicy::SizeType)3, (TQSizePolicy::SizeType)3, 0, 0, parametersFrame->sizePolicy().hasHeightForWidth() ) ); + parametersFrame->setFrameShape( TQFrame::NoFrame ); + parametersFrame->setFrameShadow( TQFrame::Plain ); + parametersFrame->setLineWidth( 0 ); + parametersFrameLayout = new TQVBoxLayout( parametersFrame, 0, 0, "parametersFrameLayout"); + + titleButton = new TQPushButton( parametersFrame, "titleButton" ); + titleButton->setToggleButton( TRUE ); + parametersFrameLayout->addWidget( titleButton ); + + titleFrame = new TQFrame( parametersFrame, "titleFrame" ); + titleFrame->setFrameShape( TQFrame::StyledPanel ); + titleFrame->setFrameShadow( TQFrame::Raised ); + titleFrameLayout = new TQVBoxLayout( titleFrame, 5, 3, "titleFrameLayout"); + + requireAllTitle = new TQCheckBox( i18n("Require All Words"), titleFrame ); + titleFrameLayout->addWidget( requireAllTitle ); + + TQHBox *titleHBox = new TQHBox( titleFrame ); + /*TQLabel *titleInfoLabel = */new TQLabel(i18n("Keywords:"),titleHBox); + titleEdit = new TQLineEdit( titleHBox, "titleEdit" ); + titleFrameLayout->addWidget( titleHBox ); + + parametersFrameLayout->addWidget( titleFrame ); + titleFrameSpacer = new TQSpacerItem( 0, 0, TQSizePolicy::Minimum, TQSizePolicy::Preferred ); + parametersFrameLayout->addItem( titleFrameSpacer ); + + + ingredientButton = new TQPushButton( parametersFrame, "ingredientButton" ); + ingredientButton->setToggleButton( TRUE ); + parametersFrameLayout->addWidget( ingredientButton ); + + ingredientFrame = new TQFrame( parametersFrame, "ingredientFrame" ); + ingredientFrame->setFrameShape( TQFrame::StyledPanel ); + ingredientFrame->setFrameShadow( TQFrame::Raised ); + ingredientFrameLayout = new TQGridLayout( ingredientFrame, 1, 1, 3, 3, "ingredientFrameLayout"); + + TQLabel *ingredientInfoLabel = new TQLabel(i18n("Enter ingredients: (e.g. chicken pasta \"white wine\")"),ingredientFrame); + ingredientInfoLabel->setTextFormat( TQt::RichText ); + ingredientFrameLayout->addMultiCellWidget( ingredientInfoLabel, 0, 0, 0, 1 ); + + ingredientsAllEdit = new TQLineEdit( ingredientFrame, "ingredientsAllEdit" ); + + ingredientFrameLayout->addWidget( ingredientsAllEdit, 1, 1 ); + + ingredientsAnyEdit = new TQLineEdit( ingredientFrame, "ingredientsAnyEdit" ); + + ingredientFrameLayout->addWidget( ingredientsAnyEdit, 2, 1 ); + + textLabel1_2 = new TQLabel( ingredientFrame, "textLabel1_2" ); + + ingredientFrameLayout->addWidget( textLabel1_2, 2, 0 ); + + textLabel1 = new TQLabel( ingredientFrame, "textLabel1" ); + + ingredientFrameLayout->addWidget( textLabel1, 1, 0 ); + + ingredientsWithoutEdit = new TQLineEdit( ingredientFrame, "ingredientsWithoutEdit" ); + + ingredientFrameLayout->addWidget( ingredientsWithoutEdit, 3, 1 ); + + textLabel1_3 = new TQLabel( ingredientFrame, "textLabel1_3" ); + + ingredientFrameLayout->addWidget( textLabel1_3, 3, 0 ); + parametersFrameLayout->addWidget( ingredientFrame ); + spacer3_2_3_2_2 = new TQSpacerItem( 0, 0, TQSizePolicy::Minimum, TQSizePolicy::Preferred ); + parametersFrameLayout->addItem( spacer3_2_3_2_2 ); + + + categoriesButton = new TQPushButton( parametersFrame, "categoriesButton" ); + categoriesButton->setToggleButton( TRUE ); + parametersFrameLayout->addWidget( categoriesButton ); + + categoryFrame = new TQFrame( parametersFrame, "categoryFrame" ); + categoryFrame->setFrameShape( TQFrame::StyledPanel ); + categoryFrame->setFrameShadow( TQFrame::Raised ); + categoryFrameLayout = new TQGridLayout( categoryFrame, 1, 1, 3, 3, "categoryFrameLayout"); + + TQLabel *categoryInfoLabel = new TQLabel(i18n("Enter categories: (e.g. Desserts Pastas \"Main Dishes\")"),categoryFrame); + categoryInfoLabel->setTextFormat( TQt::RichText ); + categoryFrameLayout->addMultiCellWidget( categoryInfoLabel, 0, 0, 0, 1 ); + + categoriesAllEdit = new TQLineEdit( categoryFrame, "categoriesAllEdit" ); + + categoryFrameLayout->addWidget( categoriesAllEdit, 1, 1 ); + + textLabel1_5 = new TQLabel( categoryFrame, "textLabel1_5" ); + + categoryFrameLayout->addWidget( textLabel1_5, 1, 0 ); + + textLabel1_3_3 = new TQLabel( categoryFrame, "textLabel1_3_3" ); + + categoryFrameLayout->addWidget( textLabel1_3_3, 3, 0 ); + + categoriesAnyEdit = new TQLineEdit( categoryFrame, "categoriesAnyEdit" ); + + categoryFrameLayout->addWidget( categoriesAnyEdit, 2, 1 ); + + textLabel1_2_3 = new TQLabel( categoryFrame, "textLabel1_2_3" ); + + categoryFrameLayout->addWidget( textLabel1_2_3, 2, 0 ); + + categoriesNotEdit = new TQLineEdit( categoryFrame, "categoriesNotEdit" ); + + categoryFrameLayout->addWidget( categoriesNotEdit, 3, 1 ); + parametersFrameLayout->addWidget( categoryFrame ); + spacer3_2_3_2 = new TQSpacerItem( 0, 0, TQSizePolicy::Minimum, TQSizePolicy::Preferred ); + parametersFrameLayout->addItem( spacer3_2_3_2 ); + + + authorsButton = new TQPushButton( parametersFrame, "authorsButton" ); + authorsButton->setToggleButton( TRUE ); + parametersFrameLayout->addWidget( authorsButton ); + + authorsFrame = new TQFrame( parametersFrame, "authorsFrame" ); + authorsFrame->setFrameShape( TQFrame::StyledPanel ); + authorsFrame->setFrameShadow( TQFrame::Raised ); + authorsFrameLayout = new TQGridLayout( authorsFrame, 1, 1, 3, 3, "authorsFrameLayout"); + + TQLabel *authorsInfoLabel = new TQLabel(i18n("Enter author name (e.g. Smith or \"Jane Doe\")"),authorsFrame); + authorsInfoLabel->setTextFormat( TQt::RichText ); + authorsFrameLayout->addMultiCellWidget( authorsInfoLabel, 0, 0, 0, 1 ); + + textLabel1_2_4 = new TQLabel( authorsFrame, "textLabel1_2_4" ); + + authorsFrameLayout->addWidget( textLabel1_2_4, 1, 0 ); + + textLabel1_6 = new TQLabel( authorsFrame, "textLabel1_6" ); + + authorsFrameLayout->addWidget( textLabel1_6, 2, 0 ); + + textLabel1_3_4 = new TQLabel( authorsFrame, "textLabel1_3_4" ); + + authorsFrameLayout->addWidget( textLabel1_3_4, 3, 0 ); + + authorsAnyEdit = new TQLineEdit( authorsFrame, "authorsAnyEdit" ); + + authorsFrameLayout->addWidget( authorsAnyEdit, 1, 1 ); + + authorsAllEdit = new TQLineEdit( authorsFrame, "authorsAllEdit" ); + + authorsFrameLayout->addWidget( authorsAllEdit, 2, 1 ); + + authorsWithoutEdit = new TQLineEdit( authorsFrame, "authorsWithoutEdit" ); + + authorsFrameLayout->addWidget( authorsWithoutEdit, 3, 1 ); + parametersFrameLayout->addWidget( authorsFrame ); + spacer3_2_3 = new TQSpacerItem( 0, 0, TQSizePolicy::Minimum, TQSizePolicy::Preferred ); + parametersFrameLayout->addItem( spacer3_2_3 ); + + + servingsButton = new TQPushButton( parametersFrame, "servingsButton" ); + servingsButton->setToggleButton( TRUE ); + parametersFrameLayout->addWidget( servingsButton ); + + servingsFrame = new TQFrame( parametersFrame, "servingsFrame" ); + servingsFrame->setFrameShape( TQFrame::StyledPanel ); + servingsFrame->setFrameShadow( TQFrame::Raised ); + servingsFrameLayout = new TQVBoxLayout( servingsFrame, 3, 3, "servingsFrameLayout"); + + enableServingsCheckBox = new TQCheckBox( servingsFrame, "enableServingsCheckBox" ); + servingsFrameLayout->addWidget( enableServingsCheckBox ); + + layout5 = new TQHBoxLayout( 0, 0, 3, "layout5"); + + servingsComboBox = new TQComboBox( FALSE, servingsFrame, "servingsComboBox" ); + servingsComboBox->setEnabled( FALSE ); + servingsComboBox->setSizePolicy( TQSizePolicy( (TQSizePolicy::SizeType)1, (TQSizePolicy::SizeType)0, 1, 0, servingsComboBox->sizePolicy().hasHeightForWidth() ) ); + layout5->addWidget( servingsComboBox ); + + servingsSpinBox = new TQSpinBox( servingsFrame, "servingsSpinBox" ); + servingsSpinBox->setEnabled( FALSE ); + servingsSpinBox->setMinValue( 1 ); + servingsSpinBox->setMaxValue( 9999 ); + layout5->addWidget( servingsSpinBox ); + servingsFrameLayout->addLayout( layout5 ); + parametersFrameLayout->addWidget( servingsFrame ); + spacer3_2_2 = new TQSpacerItem( 0, 0, TQSizePolicy::Minimum, TQSizePolicy::Preferred ); + parametersFrameLayout->addItem( spacer3_2_2 ); + + + prepTimeButton = new TQPushButton( parametersFrame, "prepTimeButton" ); + prepTimeButton->setToggleButton( TRUE ); + parametersFrameLayout->addWidget( prepTimeButton ); + + prepTimeFrame = new TQFrame( parametersFrame, "prepTimeFrame" ); + prepTimeFrame->setFrameShape( TQFrame::StyledPanel ); + prepTimeFrame->setFrameShadow( TQFrame::Raised ); + prepTimeFrameLayout = new TQVBoxLayout( prepTimeFrame, 3, 3, "prepTimeFrameLayout"); + + enablePrepTimeCheckBox = new TQCheckBox( prepTimeFrame, "enablePrepTimeCheckBox" ); + prepTimeFrameLayout->addWidget( enablePrepTimeCheckBox ); + + layout6 = new TQHBoxLayout( 0, 0, 3, "layout6"); + + prepTimeComboBox = new TQComboBox( FALSE, prepTimeFrame, "prepTimeComboBox" ); + prepTimeComboBox->setEnabled( FALSE ); + prepTimeComboBox->setSizePolicy( TQSizePolicy( (TQSizePolicy::SizeType)1, (TQSizePolicy::SizeType)0, 1, 0, prepTimeComboBox->sizePolicy().hasHeightForWidth() ) ); + layout6->addWidget( prepTimeComboBox ); + + prepTimeEdit = new TQTimeEdit( prepTimeFrame, "prepTimeEdit" ); + prepTimeEdit->setEnabled( FALSE ); + prepTimeEdit->setDisplay( int( TQTimeEdit::Minutes | TQTimeEdit::Hours ) ); + layout6->addWidget( prepTimeEdit ); + prepTimeFrameLayout->addLayout( layout6 ); + parametersFrameLayout->addWidget( prepTimeFrame ); + spacer15 = new TQSpacerItem( 0, 0, TQSizePolicy::Minimum, TQSizePolicy::Preferred ); + parametersFrameLayout->addItem( spacer15 ); + + + instructionsButton = new TQPushButton( parametersFrame, "instructionsButton" ); + instructionsButton->setToggleButton( TRUE ); + parametersFrameLayout->addWidget( instructionsButton ); + + instructionsFrame = new TQFrame( parametersFrame, "instructionsFrame" ); + instructionsFrame->setFrameShape( TQFrame::StyledPanel ); + instructionsFrame->setFrameShadow( TQFrame::Raised ); + instructionsFrameLayout = new TQVBoxLayout( instructionsFrame, 5, 3, "instructionsFrameLayout"); + + requireAllInstructions = new TQCheckBox( i18n("Require All Words"), instructionsFrame ); + instructionsFrameLayout->addWidget( requireAllInstructions ); + + TQHBox *instructionsHBox = new TQHBox(instructionsFrame); + /*TQLabel *instructionsInfoLabel = */new TQLabel(i18n("Keywords:"),instructionsHBox); + + instructionsEdit = new TQLineEdit( instructionsHBox, "instructionsEdit" ); + instructionsFrameLayout->addWidget( instructionsHBox ); + + parametersFrameLayout->addWidget( instructionsFrame ); + instructionsFrameSpacer = new TQSpacerItem( 0, 0, TQSizePolicy::Minimum, TQSizePolicy::Preferred ); + parametersFrameLayout->addItem( instructionsFrameSpacer ); + + + metaDataButton = new TQPushButton( parametersFrame, "metaDataButton" ); + metaDataButton->setToggleButton( TRUE ); + parametersFrameLayout->addWidget( metaDataButton ); + + metaDataFrame = new TQFrame( parametersFrame, "metaDataFrame" ); + metaDataFrame->setFrameShape( TQFrame::StyledPanel ); + metaDataFrame->setFrameShadow( TQFrame::Raised ); + metaDataFrameLayout = new TQVBoxLayout( metaDataFrame, 5, 3, "metaDataFrameLayout"); + + TQLabel *createdLabel = new TQLabel( i18n("Created:"), metaDataFrame ); + metaDataFrameLayout->addWidget( createdLabel ); + + TQHBox *createdHBox = new TQHBox(metaDataFrame); + createdStartDateEdit = new KDateEdit(createdHBox,"createdStartEdit"); + createdStartDateEdit->setSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::Preferred ); + (void)new TQLabel(" - ",createdHBox); + createdEndDateEdit = new KDateEdit(createdHBox,"createdEndEdit"); + createdEndDateEdit->setSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::Preferred ); + metaDataFrameLayout->addWidget( createdHBox ); + + TQLabel *modifiedLabel = new TQLabel( i18n("Modified:"), metaDataFrame ); + metaDataFrameLayout->addWidget( modifiedLabel ); + + TQHBox *modifiedHBox = new TQHBox(metaDataFrame); + modifiedStartDateEdit = new KDateEdit(modifiedHBox,"modifiedStartEdit"); + modifiedStartDateEdit->setSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::Preferred ); + (void)new TQLabel(" - ",modifiedHBox); + modifiedEndDateEdit = new KDateEdit(modifiedHBox,"modifiedEndEdit"); + modifiedEndDateEdit->setSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::Preferred ); + metaDataFrameLayout->addWidget( modifiedHBox ); + + TQLabel *accessedLabel = new TQLabel( i18n("Last Accessed:"), metaDataFrame ); + metaDataFrameLayout->addWidget( accessedLabel ); + + TQHBox *accessedHBox = new TQHBox(metaDataFrame); + accessedStartDateEdit = new KDateEdit(accessedHBox,"accessedStartEdit"); + accessedStartDateEdit->setSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::Preferred ); + (void)new TQLabel(" - ",accessedHBox); + accessedEndDateEdit = new KDateEdit(accessedHBox,"accessedEndEdit"); + accessedEndDateEdit->setSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::Preferred ); + metaDataFrameLayout->addWidget( accessedHBox ); + + parametersFrameLayout->addWidget( metaDataFrame ); + metaDataFrameSpacer = new TQSpacerItem( 0, 0, TQSizePolicy::Minimum, TQSizePolicy::Preferred ); + parametersFrameLayout->addItem( metaDataFrameSpacer ); + + //=============RATINGS FRAME===========// + ratingsButton = new TQPushButton( parametersFrame, "ratingsButton" ); + ratingsButton->setToggleButton( TRUE ); + parametersFrameLayout->addWidget( ratingsButton ); + + ratingButtonGroup = new TQButtonGroup( parametersFrame, "ratingButtonGroup" ); + ratingButtonGroup->setLineWidth( 0 ); + ratingButtonGroup->setColumnLayout(0, TQt::Vertical ); + ratingButtonGroup->layout()->setSpacing( 5 ); + ratingButtonGroup->layout()->setMargin( 3 ); + ratingButtonGroupLayout = new TQVBoxLayout( ratingButtonGroup->layout() ); + ratingButtonGroupLayout->setAlignment( TQt::AlignTop ); + + ratingAvgRadioButton = new TQRadioButton( ratingButtonGroup, "ratingAvgRadioButton" ); + ratingAvgRadioButton->setChecked( TRUE ); + ratingButtonGroupLayout->addWidget( ratingAvgRadioButton ); + + ratingAvgFrame = new TQFrame( ratingButtonGroup, "ratingAvgFrame" ); + ratingAvgFrame->setLineWidth( 0 ); + ratingAvgFrameLayout = new TQHBoxLayout( ratingAvgFrame, 2, 2, "ratingAvgFrameLayout"); + + avgStarsEdit = new FractionInput( ratingAvgFrame ); + avgStarsEdit->setAllowRange(true); + ratingAvgFrameLayout->addWidget( avgStarsEdit ); + + avgStarsLabel = new TQLabel( ratingAvgFrame, "avgStarsLabel" ); + ratingAvgFrameLayout->addWidget( avgStarsLabel ); + ratingButtonGroupLayout->addWidget( ratingAvgFrame ); + + criterionRadioButton = new TQRadioButton( ratingButtonGroup, "criterionRadioButton" ); + ratingButtonGroupLayout->addWidget( criterionRadioButton ); + + criterionFrame = new TQFrame( ratingButtonGroup, "criterionFrame" ); + criterionFrame->setEnabled( FALSE ); + criterionFrame->setLineWidth( 0 ); + criterionFrameLayout = new TQVBoxLayout( criterionFrame, 2, 2, "criterionFrameLayout"); + + layout12 = new TQHBoxLayout( 0, 0, 6, "layout12"); + + criteriaComboBox = new CriteriaComboBox( FALSE, criterionFrame, database ); + criteriaComboBox->setEditable( false ); + criteriaComboBox->reload(); + layout12->addWidget( criteriaComboBox ); + + starsWidget = new FractionInput( criterionFrame ); + starsWidget->setAllowRange(true); + starsWidget->setSizePolicy( TQSizePolicy::Ignored, TQSizePolicy::Preferred ); + layout12->addWidget( starsWidget ); + + + addCriteriaButton = new TQPushButton( criterionFrame, "addCriteriaButton" ); + addCriteriaButton->setSizePolicy( TQSizePolicy( (TQSizePolicy::SizeType)0, (TQSizePolicy::SizeType)0, 0, 0, addCriteriaButton->sizePolicy().hasHeightForWidth() ) ); + addCriteriaButton->setMaximumSize( TQSize( 30, 30 ) ); + layout12->addWidget( addCriteriaButton ); +#if 0 + removeCriteriaButton = new TQPushButton( criterionFrame, "removeCriteriaButton" ); + removeCriteriaButton->setSizePolicy( TQSizePolicy( (TQSizePolicy::SizeType)0, (TQSizePolicy::SizeType)0, 0, 0, removeCriteriaButton->sizePolicy().hasHeightForWidth() ) ); + removeCriteriaButton->setMaximumSize( TQSize( 30, 30 ) ); + layout12->addWidget( removeCriteriaButton ); +#endif + criterionFrameLayout->addLayout( layout12 ); + + criteriaListView = new TDEListView( criterionFrame, "criteriaListView" ); + criteriaListView->setSizePolicy( TQSizePolicy::Preferred, TQSizePolicy::Minimum ); + criteriaListView->addColumn( i18n( "Criterion" ) ); + criteriaListView->addColumn( i18n( "Stars" ) ); + criterionFrameLayout->addWidget( criteriaListView ); + ratingButtonGroupLayout->addWidget( criterionFrame ); + + parametersFrameLayout->addWidget( ratingButtonGroup ); + ratingsFrameSpacer = new TQSpacerItem( 0, 0, TQSizePolicy::Minimum, TQSizePolicy::Expanding ); + parametersFrameLayout->addItem( ratingsFrameSpacer ); + + + scrollView1->addChild( parametersFrame ); + layout7->addWidget( scrollView1 ); + + layout9 = new TQHBoxLayout( 0, 0, 3, "layout9"); + + clearButton = new KPushButton( this, "clearButton" ); + clearButton->setSizePolicy( TQSizePolicy( (TQSizePolicy::SizeType)1, (TQSizePolicy::SizeType)0, 0, 0, clearButton->sizePolicy().hasHeightForWidth() ) ); + layout9->addWidget( clearButton ); + spacer3 = new TQSpacerItem( 110, 0, TQSizePolicy::Expanding, TQSizePolicy::Minimum ); + layout9->addItem( spacer3 ); + + findButton = new KPushButton( this, "findButton" ); + findButton->setSizePolicy( TQSizePolicy( (TQSizePolicy::SizeType)1, (TQSizePolicy::SizeType)0, 0, 0, findButton->sizePolicy().hasHeightForWidth() ) ); + layout9->addWidget( findButton ); + layout7->addLayout( layout9 ); + AdvancedSearchDialogLayout->addLayout( layout7 ); + + resultsListView = new TDEListView( this, "resultsListView" ); + resultsListView->setSelectionMode( TQListView::Extended ); + resultsListView->setSizePolicy( TQSizePolicy( (TQSizePolicy::SizeType)7, (TQSizePolicy::SizeType)7, 0, 1, resultsListView->sizePolicy().hasHeightForWidth() ) ); + AdvancedSearchDialogLayout->addWidget( resultsListView ); + languageChange(); + clearWState( WState_Polished ); + /// + ///END OF AUTOMATICALLY GENERATED GUI CODE/// + /// + + resultsListView->setSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::Preferred ); + scrollView1->setSizePolicy( TQSizePolicy::Preferred, TQSizePolicy::Ignored ); + + AdvancedSearchDialogLayout->setStretchFactor( resultsListView, 2 ); + + scrollView1->setHScrollBarMode( TQScrollView::AlwaysOff ); + scrollView1->setResizePolicy( TQScrollView::AutoOneFit ); + + TDEConfig *config = TDEGlobal::config(); + config->setGroup( "Advanced" ); + bool show_id = config->readBoolEntry( "ShowID", false ); + resultsListView->addColumn( i18n( "Title" ) ); + resultsListView->addColumn( i18n( "Id" ), show_id ? -1 : 0 ); + + actionHandler = new RecipeActionsHandler( resultsListView, database, RecipeActionsHandler::Open | RecipeActionsHandler::Edit | RecipeActionsHandler::Export | RecipeActionsHandler::CopyToClipboard | RecipeActionsHandler::Remove ); + + connect( titleEdit, TQ_SIGNAL( returnPressed() ), TQ_SLOT( search() ) ); + connect( ingredientsAllEdit, TQ_SIGNAL( returnPressed() ), TQ_SLOT( search() ) ); + connect( ingredientsAnyEdit, TQ_SIGNAL( returnPressed() ), TQ_SLOT( search() ) ); + connect( ingredientsWithoutEdit, TQ_SIGNAL( returnPressed() ), TQ_SLOT( search() ) ); + connect( authorsAnyEdit, TQ_SIGNAL( returnPressed() ), TQ_SLOT( search() ) ); + connect( authorsAllEdit, TQ_SIGNAL( returnPressed() ), TQ_SLOT( search() ) ); + connect( authorsWithoutEdit, TQ_SIGNAL( returnPressed() ), TQ_SLOT( search() ) ); + connect( categoriesNotEdit, TQ_SIGNAL( returnPressed() ), TQ_SLOT( search() ) ); + connect( categoriesAnyEdit, TQ_SIGNAL( returnPressed() ), TQ_SLOT( search() ) ); + connect( categoriesAllEdit, TQ_SIGNAL( returnPressed() ), TQ_SLOT( search() ) ); + connect( ingredientsWithoutEdit, TQ_SIGNAL( returnPressed() ), TQ_SLOT( search() ) ); + connect( instructionsEdit, TQ_SIGNAL( returnPressed() ), TQ_SLOT( search() ) ); + + connect( findButton, TQ_SIGNAL( clicked() ), TQ_SLOT( search() ) ); + connect( clearButton, TQ_SIGNAL( clicked() ), TQ_SLOT( clear() ) ); + + connect( enableServingsCheckBox, TQ_SIGNAL( toggled( bool ) ), servingsSpinBox, TQ_SLOT( setEnabled( bool ) ) ); + connect( enableServingsCheckBox, TQ_SIGNAL( toggled( bool ) ), servingsComboBox, TQ_SLOT( setEnabled( bool ) ) ); + connect( enablePrepTimeCheckBox, TQ_SIGNAL( toggled( bool ) ), prepTimeEdit, TQ_SLOT( setEnabled( bool ) ) ); + connect( enablePrepTimeCheckBox, TQ_SIGNAL( toggled( bool ) ), prepTimeComboBox, TQ_SLOT( setEnabled( bool ) ) ); + + connect( titleButton, TQ_SIGNAL( toggled( bool ) ), titleFrame, TQ_SLOT( setShown( bool ) ) ); + connect( ingredientButton, TQ_SIGNAL( toggled( bool ) ), ingredientFrame, TQ_SLOT( setShown( bool ) ) ); + connect( authorsButton, TQ_SIGNAL( toggled( bool ) ), authorsFrame, TQ_SLOT( setShown( bool ) ) ); + connect( categoriesButton, TQ_SIGNAL( toggled( bool ) ), categoryFrame, TQ_SLOT( setShown( bool ) ) ); + connect( servingsButton, TQ_SIGNAL( toggled( bool ) ), servingsFrame, TQ_SLOT( setShown( bool ) ) ); + connect( prepTimeButton, TQ_SIGNAL( toggled( bool ) ), prepTimeFrame, TQ_SLOT( setShown( bool ) ) ); + connect( instructionsButton, TQ_SIGNAL( toggled( bool ) ), instructionsFrame, TQ_SLOT( setShown( bool ) ) ); + connect( metaDataButton, TQ_SIGNAL( toggled( bool ) ), metaDataFrame, TQ_SLOT( setShown( bool ) ) ); + connect( ratingsButton, TQ_SIGNAL( toggled( bool ) ), ratingButtonGroup, TQ_SLOT( setShown( bool ) ) ); + + connect( titleButton, TQ_SIGNAL( toggled( bool ) ), TQ_SLOT( buttonSwitched() ) ); + connect( ingredientButton, TQ_SIGNAL( toggled( bool ) ), TQ_SLOT( buttonSwitched() ) ); + connect( authorsButton, TQ_SIGNAL( toggled( bool ) ), TQ_SLOT( buttonSwitched() ) ); + connect( categoriesButton, TQ_SIGNAL( toggled( bool ) ), TQ_SLOT( buttonSwitched() ) ); + connect( servingsButton, TQ_SIGNAL( toggled( bool ) ), TQ_SLOT( buttonSwitched() ) ); + connect( prepTimeButton, TQ_SIGNAL( toggled( bool ) ), TQ_SLOT( buttonSwitched() ) ); + connect( instructionsButton, TQ_SIGNAL( toggled( bool ) ), TQ_SLOT( buttonSwitched() ) ); + connect( metaDataButton, TQ_SIGNAL( toggled( bool ) ), TQ_SLOT( buttonSwitched() ) ); + connect( ratingsButton, TQ_SIGNAL( toggled( bool ) ), TQ_SLOT( buttonSwitched() ) ); + + connect( ratingButtonGroup, TQ_SIGNAL( clicked( int ) ), this, TQ_SLOT( activateRatingOption( int ) ) ); + + connect( addCriteriaButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( slotAddRatingCriteria() ) ); + + titleFrame->setShown(false); + ingredientFrame->setShown(false); + authorsFrame->setShown(false); + categoryFrame->setShown(false); + servingsFrame->setShown(false); + prepTimeFrame->setShown(false); + instructionsFrame->setShown(false); + metaDataFrame->setShown(false); + ratingButtonGroup->setShown(false); + + connect( actionHandler, TQ_SIGNAL( recipeSelected( int, int ) ), TQ_SIGNAL( recipeSelected( int, int ) ) ); + connect( actionHandler, TQ_SIGNAL( recipesSelected( const TQValueList<int> &, int ) ), TQ_SIGNAL( recipesSelected( const TQValueList<int> &, int ) ) ); + + connect( database, TQ_SIGNAL( recipeRemoved( int ) ), TQ_SLOT( removeRecipe( int ) ) ); + + clear(); + + TDEPopupMenu *kpop = new TDEPopupMenu( criteriaListView ); + kpop->insertItem( i18n( "&Delete" ), this, TQ_SLOT( slotRemoveRatingCriteria() ), Key_Delete ); +} + +AdvancedSearchDialog::~AdvancedSearchDialog() +{} + +void AdvancedSearchDialog::languageChange() +{ + titleButton->setText( TQString("%1 >>").arg(i18n("Title")) ); + textLabel1_4->setText( i18n( "Search using the following criteria:" ) ); + ingredientButton->setText( TQString("%1 >>").arg(i18n("Ingredients")) ); + textLabel1_2->setText( i18n( "Uses any of:" ) ); + textLabel1->setText( i18n( "Uses all:" ) ); + textLabel1_3->setText( i18n( "Without:" ) ); + categoriesButton->setText( i18n( "Categories >>" ) ); + textLabel1_5->setText( i18n( "In all:" ) ); + textLabel1_3_3->setText( i18n( "Not in:" ) ); + textLabel1_2_3->setText( i18n( "In any of:" ) ); + authorsButton->setText( TQString("%1 >>").arg(i18n("Authors")) ); + textLabel1_2_4->setText( i18n( "By any of:" ) ); + textLabel1_6->setText( i18n( "By all:" ) ); + textLabel1_3_4->setText( i18n( "Not by:" ) ); + servingsButton->setText( TQString("%1 >>").arg(i18n("Yield")) ); + enableServingsCheckBox->setText( i18n( "Enabled" ) ); + servingsComboBox->clear(); + servingsComboBox->insertItem( i18n( "Yields at least:" ) ); + servingsComboBox->insertItem( i18n( "Yields at most:" ) ); + servingsComboBox->insertItem( i18n( "Yields about:" ) ); + prepTimeButton->setText( TQString("%1 >>").arg(i18n("Preparation Time")) ); + enablePrepTimeCheckBox->setText( i18n( "Enabled" ) ); + prepTimeComboBox->clear(); + prepTimeComboBox->insertItem( i18n( "Ready in at most:" ) ); + prepTimeComboBox->insertItem( i18n( "Ready in about:" ) ); + instructionsButton->setText( TQString("%1 >>").arg(i18n("Instructions")) ); + metaDataButton->setText( TQString("%1 >>").arg(i18n("Meta Data")) ); + clearButton->setText( i18n( "C&lear" ) ); + findButton->setText( i18n( "&Search" ) ); + ratingAvgRadioButton->setText( i18n( "By average:" ) ); + avgStarsLabel->setText( i18n( "stars" ) ); + criterionRadioButton->setText( i18n( "By criteria:" ) ); + addCriteriaButton->setText( i18n( "+" ) ); + //removeCriteriaButton->setText( i18n( "-" ) ); + criteriaListView->header()->setLabel( 0, i18n( "Criteria" ) ); + criteriaListView->header()->setLabel( 1, i18n( "Stars" ) ); + ratingsButton->setText( TQString("%1 >>").arg(i18n("Ratings")) ); +} + +void AdvancedSearchDialog::removeRecipe( int id ) +{ + TQListViewItemIterator iterator( resultsListView ); + while ( iterator.current() ) { + if ( iterator.current()->rtti() == 1000 ) { + RecipeListItem * recipe_it = ( RecipeListItem* ) iterator.current(); + if ( recipe_it->recipeID() == id ) + delete recipe_it; + } + ++iterator; + } +} + +void AdvancedSearchDialog::clear() +{ + resultsListView->clear(); + authorsAllEdit->clear(); + authorsWithoutEdit->clear(); + authorsAnyEdit->clear(); + categoriesAllEdit->clear(); + categoriesNotEdit->clear(); + categoriesAnyEdit->clear(); + ingredientsAllEdit->clear(); + ingredientsWithoutEdit->clear(); + ingredientsAnyEdit->clear(); + titleEdit->clear(); + instructionsEdit->clear(); + + createdStartDateEdit->setDate( TQDate() ); + createdEndDateEdit->setDate( TQDate() ); + modifiedStartDateEdit->setDate( TQDate() ); + modifiedEndDateEdit->setDate( TQDate() ); + accessedStartDateEdit->setDate( TQDate() ); + accessedEndDateEdit->setDate( TQDate() ); + + servingsSpinBox->setValue( 1 ); + prepTimeEdit->setTime( TQTime(0,0) ); + + enablePrepTimeCheckBox->setChecked(false); + enableServingsCheckBox->setChecked(false); + + requireAllTitle->setChecked(false); + requireAllInstructions->setChecked(false); + + ratingAvgRadioButton->setChecked(true); + activateRatingOption(0); + avgStarsEdit->clear(); + criteriaListView->clear(); + starsWidget->clear(); +} + +void AdvancedSearchDialog::activateRatingOption( int button_id ) +{ + switch ( button_id ) { + case 0: + criterionFrame->setEnabled( false ); + ratingAvgFrame->setEnabled( true ); + break; + case 1: + criterionFrame->setEnabled( true ); + ratingAvgFrame->setEnabled( false ); + break; + default: + break; + } +} + +void AdvancedSearchDialog::buttonSwitched() +{ + const TQObject *sent = sender(); + + if ( sent->inherits("TQPushButton") ) { + TQPushButton *pushed = (TQPushButton*) sent; + + TQString suffix = ( pushed->state() == TQButton::On ) ? " <<" : " >>"; + pushed->setText( pushed->text().left( pushed->text().length() - 3 ) + suffix ); + } +} + +void AdvancedSearchDialog::search() +{ + TDEApplication::setOverrideCursor( KCursor::waitCursor() ); + + //we need to load more than just the title because we'll be doing further refining of the search + int load_items = RecipeDB::Title | RecipeDB::NamesOnly | RecipeDB::Noatime; + if ( !authorsAllEdit->text().isEmpty() || !authorsWithoutEdit->text().isEmpty() ) + load_items |= RecipeDB::Authors; + if ( !ingredientsAllEdit->text().isEmpty() || !ingredientsWithoutEdit->text().isEmpty() ) + load_items |= RecipeDB::Ingredients; + if ( !categoriesAllEdit->text().isEmpty() || !categoriesNotEdit->text().isEmpty() ) + load_items |= RecipeDB::Categories; + if ( (ratingAvgRadioButton->isChecked() && !avgStarsEdit->isEmpty()) || (criterionRadioButton->isChecked() && criteriaListView->firstChild()) ) + load_items |= RecipeDB::Ratings; + + RecipeSearchParameters parameters; + + parameters.titleKeywords = split(titleEdit->text(),true); + parameters.requireAllTitleWords = requireAllTitle->isChecked(); + + parameters.instructionsKeywords = split(instructionsEdit->text(),true); + parameters.requireAllInstructionsWords = requireAllInstructions->isChecked(); + + parameters.ingsOr = split(ingredientsAnyEdit->text(),true); + parameters.catsOr = split(categoriesAnyEdit->text(),true); + parameters.authorsOr = split(authorsAnyEdit->text(),true); + + if ( enablePrepTimeCheckBox->isChecked() ) + parameters.prep_time = prepTimeEdit->time(); + parameters.prep_param = prepTimeComboBox->currentItem(); + + if ( enableServingsCheckBox->isChecked() ) + parameters.servings = servingsSpinBox->value(); + parameters.servings_param = servingsComboBox->currentItem(); + + parameters.createdDateBegin = createdStartDateEdit->date(); + parameters.createdDateEnd = createdEndDateEdit->date(); + if ( parameters.createdDateEnd.date().isValid() ) + parameters.createdDateEnd = parameters.createdDateEnd.addDays(1); //we want to include the given day in the search + + parameters.modifiedDateBegin = modifiedStartDateEdit->date(); + parameters.modifiedDateEnd = modifiedEndDateEdit->date(); + if ( parameters.modifiedDateEnd.date().isValid() ) + parameters.modifiedDateEnd = parameters.modifiedDateEnd.addDays(1); //we want to include the given day in the search + + parameters.accessedDateBegin = accessedStartDateEdit->date(); + parameters.accessedDateEnd = accessedEndDateEdit->date(); + if ( parameters.accessedDateEnd.date().isValid() ) + parameters.accessedDateEnd = parameters.accessedDateEnd.addDays(1); //we want to include the given day in the search + + START_TIMER("Doing database SQL search"); + RecipeList allRecipes; + database->search( &allRecipes, load_items, parameters ); + END_TIMER(); + + /* + * Ideally, would be done by the above SQL query, but I have no idea how to accomplish this. + */ + + START_TIMER("Further narrowing the search (no SQL)"); + TQStringList items = split(authorsAllEdit->text()); + for ( TQStringList::const_iterator author_it = items.begin(); author_it != items.end(); ++author_it ) { + for ( RecipeList::iterator it = allRecipes.begin(); it != allRecipes.end(); ++it ) { + if ( ( *it ).authorList.findByName( TQRegExp(*author_it,false, true) ).id == -1 ) { + it = allRecipes.remove( it ); + it--; + } + } + } + items = split(authorsWithoutEdit->text()); + for ( TQStringList::const_iterator author_it = items.begin(); author_it != items.end(); ++author_it ) { + for ( RecipeList::iterator it = allRecipes.begin(); it != allRecipes.end(); ++it ) { + if ( ( *it ).authorList.findByName( TQRegExp(*author_it,false,true) ).id != -1 ) { + it = allRecipes.remove( it ); + it--; + } + } + } + + //narrow down by categories + items = split(categoriesAllEdit->text()); + for ( TQStringList::const_iterator cat_it = items.begin(); cat_it != items.end(); ++cat_it ) { + for ( RecipeList::iterator it = allRecipes.begin(); it != allRecipes.end(); ++it ) { + if ( ( *it ).categoryList.findByName( TQRegExp(*cat_it,false,true) ).id == -1 ) { + it = allRecipes.remove( it ); + it--; + } + } + } + items = split(categoriesNotEdit->text()); + for ( TQStringList::const_iterator cat_it = items.begin(); cat_it != items.end(); ++cat_it ) { + for ( RecipeList::iterator it = allRecipes.begin(); it != allRecipes.end(); ++it ) { + if ( ( *it ).categoryList.findByName( TQRegExp(*cat_it,false,true) ).id != -1 ) { + it = allRecipes.remove( it ); + it--; + } + } + } + + //narrow down by ingredients + items = split(ingredientsAllEdit->text()); + for ( TQStringList::const_iterator ing_it = items.begin(); ing_it != items.end(); ++ing_it ) { + for ( RecipeList::iterator it = allRecipes.begin(); it != allRecipes.end(); ++it ) { + if ( ( *it ).ingList.findByName( TQRegExp(*ing_it,false,true) ).ingredientID == -1 ) { + it = allRecipes.remove( it ); + it--; + } + } + } + items = split(ingredientsWithoutEdit->text()); + for ( TQStringList::const_iterator ing_it = items.begin(); ing_it != items.end(); ++ing_it ) { + for ( RecipeList::iterator it = allRecipes.begin(); it != allRecipes.end(); ++it ) { + if ( ( *it ).ingList.findByName( TQRegExp(*ing_it,false,true) ).ingredientID != -1 ) { + it = allRecipes.remove( it ); + it--; + } + } + } + + if ( ratingAvgRadioButton->isChecked() && !avgStarsEdit->isEmpty() ) { + for ( RecipeList::iterator recipe_it = allRecipes.begin(); recipe_it != allRecipes.end(); ++recipe_it ) { + double sum = 0; + int count = 0; + + for ( RatingList::iterator rating_it = (*recipe_it).ratingList.begin(); rating_it != (*recipe_it).ratingList.end(); ++rating_it ) { + sum += (*rating_it).average(); + ++count; + } + + if ( count != 0 ) { + double average = sum/count; + + double stars; + double stars_offset; + avgStarsEdit->value(stars,stars_offset); + if ( stars_offset < 1e-10 ) { //if an exact amount is given, search for an amount within 0.5 of what is given + //we can get negatives here, but it really doesn't matter + stars = stars-0.5; + stars_offset = 1.0; + } + + + kdDebug()<<"average for "<<(*recipe_it).title<<" "<<average<<endl; + if ( average < stars || average > stars + stars_offset ) { + recipe_it = allRecipes.remove( recipe_it ); + recipe_it--; + } + } + else { + recipe_it = allRecipes.remove( recipe_it ); + recipe_it--; + } + } + } + + //TODO: Clean this up and/or do it more efficiently + if ( criterionRadioButton->isChecked() && criteriaListView->firstChild() ) { + for ( RecipeList::iterator recipe_it = allRecipes.begin(); recipe_it != allRecipes.end(); ++recipe_it ) { + TQMap< int, double > idSumMap; + TQMap< int, int > idCountMap; + + for ( RatingList::const_iterator rating_it = (*recipe_it).ratingList.begin(); rating_it != (*recipe_it).ratingList.end(); ++rating_it ) { + for ( RatingCriteriaList::const_iterator rc_it = (*rating_it).ratingCriteriaList.begin(); rc_it != (*rating_it).ratingCriteriaList.end(); ++rc_it ) { + TQMap< int, double >::iterator sum_it = idSumMap.find((*rc_it).id); + if ( sum_it == idSumMap.end() ) + sum_it = idSumMap.insert((*rc_it).id,0); + (*sum_it) += (*rc_it).stars; + + TQMap< int, int >::iterator count_it = idCountMap.find((*rc_it).id); + if ( count_it == idCountMap.end() ) + count_it = idCountMap.insert((*rc_it).id,0); + (*count_it)++; + } + } + + for ( TQListViewItem *item = criteriaListView->firstChild(); item; item = item->nextSibling() ) { + Ingredient i; i.setAmount( item->text(1) ); + double stars = i.amount; + double stars_offset = i.amount_offset; + + if ( stars_offset < 1e-10 ) { //if an exact amount is given, search for an amount within 0.5 of what is given + //we can get negatives here, but it really doesn't matter + stars = stars-0.5; + stars_offset = 1.0; + } + + int id = item->text(2).toInt(); + + TQMap< int, double >::iterator sum_it = idSumMap.find(id); + if ( sum_it != idSumMap.end() ) { + TQMap< int, int >::iterator count_it = idCountMap.find(id); + double average = (*sum_it)/(*count_it); + + if ( average < stars || average > stars + stars_offset ) { + recipe_it = allRecipes.remove( recipe_it ); + recipe_it--; + break; + } + } + else { + recipe_it = allRecipes.remove( recipe_it ); + recipe_it--; + break; + } + } + } + } + END_TIMER(); + + + //now display the recipes left + resultsListView->clear(); + for ( RecipeList::const_iterator it = allRecipes.begin(); it != allRecipes.end(); ++it ) { + ( void ) new RecipeListItem( resultsListView, *it ); + } + + if ( !resultsListView->firstChild() ) { + ( void ) new TQListViewItem( resultsListView, "--- "+i18n("No matching recipes found")+" ---"); + } + + TDEApplication::restoreOverrideCursor(); +} + +TQStringList AdvancedSearchDialog::split( const TQString &text, bool sql_wildcards ) const +{ + TQStringList result; + + // To keep quoted words together, first split on quotes, + // and then split again on the even numbered items + + TQStringList temp = TQStringList::split('"',text,true); + for ( uint i = 0; i < temp.count(); ++i ) { + if ( i & 1 ) //odd + result += temp[i].stripWhiteSpace(); + else //even + result += TQStringList::split(' ',temp[i]); + } + + if ( sql_wildcards ) { + for ( TQStringList::iterator it = result.begin(); it != result.end(); ++it ) { + (*it).replace("%","\\%"); + (*it).replace("_","\\_"); + + (*it).replace("*","%"); + (*it).replace("?","_"); + } + } + + return result; +} + +void AdvancedSearchDialog::slotAddRatingCriteria() +{ + TQListViewItem * it = new TQListViewItem(criteriaListView,criteriaComboBox->currentText()); + + MixedNumber stars; + double stars_offset; + starsWidget->value(stars,stars_offset); + TQString stars_str = stars.toString(); + if ( stars_offset > 0 ) + stars_str += "-"+MixedNumber( stars + stars_offset ).toString(); + else if ( stars.toDouble() <= 1e-10 ) + stars_str = ""; + + it->setText(1,stars_str); + it->setText(2,TQString::number(criteriaComboBox->criteriaID(criteriaComboBox->currentItem()))); +} + +void AdvancedSearchDialog::slotRemoveRatingCriteria() +{ + delete criteriaListView->selectedItem(); +} + +#include "advancedsearchdialog.moc" diff --git a/src/dialogs/advancedsearchdialog.h b/src/dialogs/advancedsearchdialog.h new file mode 100644 index 0000000..65eb171 --- /dev/null +++ b/src/dialogs/advancedsearchdialog.h @@ -0,0 +1,179 @@ +/*************************************************************************** +* Copyright (C) 2003-2004 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef ADVANCEDSEARCHDIALOG_H +#define ADVANCEDSEARCHDIALOG_H + +#include <tqwidget.h> + +class TQVBoxLayout; +class TQHBoxLayout; +class TQGridLayout; +class TQSpacerItem; +class TQScrollView; +class TQPushButton; +class TQFrame; +class TQLineEdit; +class TQLabel; +class TQCheckBox; +class TQComboBox; +class TQSpinBox; +class TQTimeEdit; +class KPushButton; +class TDEListView; +class TQListViewItem; +class TQRadioButton; +class TQButtonGroup; + +class KDateEdit; +class RecipeDB; +class RecipeActionsHandler; +class FractionInput; +class CriteriaComboBox; + + +class AdvancedSearchDialog : public TQWidget +{ + TQ_OBJECT + +public: + AdvancedSearchDialog( TQWidget *parent, RecipeDB * ); + ~AdvancedSearchDialog(); + + virtual void languageChange(); + + RecipeActionsHandler *actionHandler; + +protected: + TQLabel* textLabel1_4; + TQScrollView* scrollView1; + TQFrame* parametersFrame; + TQLineEdit* titleEdit; + TQPushButton* titleButton; + TQFrame* titleFrame; + TQPushButton* ingredientButton; + TQFrame* ingredientFrame; + TQLineEdit* ingredientsAllEdit; + TQLineEdit* ingredientsAnyEdit; + TQLabel* textLabel1_2; + TQLabel* textLabel1; + TQLineEdit* ingredientsWithoutEdit; + TQLabel* textLabel1_3; + TQPushButton* categoriesButton; + TQFrame* categoryFrame; + TQLineEdit* categoriesAllEdit; + TQLabel* textLabel1_5; + TQLabel* textLabel1_3_3; + TQLineEdit* categoriesAnyEdit; + TQLabel* textLabel1_2_3; + TQLineEdit* categoriesNotEdit; + TQPushButton* authorsButton; + TQFrame* authorsFrame; + TQLabel* textLabel1_2_4; + TQLabel* textLabel1_6; + TQLabel* textLabel1_3_4; + TQLineEdit* authorsAnyEdit; + TQLineEdit* authorsAllEdit; + TQLineEdit* authorsWithoutEdit; + TQPushButton* servingsButton; + TQFrame* servingsFrame; + TQCheckBox* enableServingsCheckBox; + TQComboBox* servingsComboBox; + TQSpinBox* servingsSpinBox; + TQPushButton* prepTimeButton; + TQFrame* prepTimeFrame; + TQCheckBox* enablePrepTimeCheckBox; + TQComboBox* prepTimeComboBox; + TQTimeEdit* prepTimeEdit; + TQLineEdit* instructionsEdit; + TQPushButton* instructionsButton; + TQFrame* instructionsFrame; + KPushButton* clearButton; + KPushButton* findButton; + TDEListView* resultsListView; + TQCheckBox *requireAllTitle; + TQCheckBox *requireAllInstructions; + KDateEdit *createdStartDateEdit; + KDateEdit *createdEndDateEdit; + KDateEdit *modifiedStartDateEdit; + KDateEdit *modifiedEndDateEdit; + KDateEdit *accessedStartDateEdit; + KDateEdit *accessedEndDateEdit; + TQPushButton* metaDataButton; + TQFrame* metaDataFrame; + TQRadioButton* ratingAvgRadioButton; + FractionInput* avgStarsEdit; + TQLabel* avgStarsLabel; + TQRadioButton* criterionRadioButton; + CriteriaComboBox* criteriaComboBox; + FractionInput* starsWidget; + TQPushButton* addCriteriaButton; + TQPushButton* removeCriteriaButton; + TDEListView* criteriaListView; + TQPushButton* ratingsButton; + TQButtonGroup *ratingButtonGroup; + TQFrame *ratingAvgFrame; + TQFrame *criterionFrame; + TQVBoxLayout *criterionFrameLayout; + TQHBoxLayout *ratingAvgFrameLayout; + TQVBoxLayout *ratingButtonGroupLayout; + + TQHBoxLayout* AdvancedSearchDialogLayout; + TQVBoxLayout* layout7; + TQHBoxLayout* scrollView1Layout; + TQVBoxLayout* parametersFrameLayout; + TQSpacerItem* spacer3_2_3_2_2; + TQSpacerItem* spacer3_2_3_2; + TQSpacerItem* spacer3_2_3; + TQSpacerItem* spacer3_2_2; + TQSpacerItem* titleFrameSpacer; + TQSpacerItem* instructionsFrameSpacer; + TQSpacerItem* metaDataFrameSpacer; + TQSpacerItem* spacer15; + TQVBoxLayout* titleFrameLayout; + TQGridLayout* ingredientFrameLayout; + TQGridLayout* categoryFrameLayout; + TQGridLayout* authorsFrameLayout; + TQVBoxLayout* servingsFrameLayout; + TQHBoxLayout* layout5; + TQVBoxLayout* prepTimeFrameLayout; + TQVBoxLayout* instructionsFrameLayout; + TQVBoxLayout* metaDataFrameLayout; + TQHBoxLayout* layout6; + TQHBoxLayout* layout9; + TQSpacerItem* spacer3; + TQVBoxLayout* ratingsFrameLayout; + TQHBoxLayout* layout11; + TQHBoxLayout* layout12; + TQSpacerItem* ratingsFrameSpacer; + + RecipeDB *database; + +signals: + void recipeSelected( int, int ); + void recipesSelected( const TQValueList<int> &, int ); + +private slots: + void search(); + void clear(); + void buttonSwitched(); + void activateRatingOption( int button_id ); + void slotAddRatingCriteria(); + void slotRemoveRatingCriteria(); + + //called by a signal from the database when a recipe is removed + void removeRecipe( int id ); + +private: + TQStringList split( const TQString &text, bool sql_wildcards = false ) const; +}; + +#endif //ADVANCEDSEARCHDIALOG_H + diff --git a/src/dialogs/authorsdialog.cpp b/src/dialogs/authorsdialog.cpp new file mode 100644 index 0000000..0a53899 --- /dev/null +++ b/src/dialogs/authorsdialog.cpp @@ -0,0 +1,66 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "authorsdialog.h" +#include "createelementdialog.h" +#include "backends/recipedb.h" +#include "widgets/authorlistview.h" + +#include <kdialog.h> +#include <tdelocale.h> +#include <tdemessagebox.h> + +AuthorsDialog::AuthorsDialog( TQWidget* parent, RecipeDB *db ) : TQWidget( parent ) +{ + + // Store pointer to database + database = db; + + TQHBoxLayout* layout = new TQHBoxLayout( this, KDialog::marginHint(), KDialog::spacingHint() ); + + //Author List + authorListView = new StdAuthorListView( this, database, true ); + layout->addWidget( authorListView ); + + //Buttons + TQVBoxLayout* vboxl = new TQVBoxLayout( KDialog::spacingHint() ); + + newAuthorButton = new TQPushButton( this ); + newAuthorButton->setText( i18n( "Create ..." ) ); + newAuthorButton->setFlat( true ); + vboxl->addWidget( newAuthorButton ); + + removeAuthorButton = new TQPushButton( this ); + removeAuthorButton->setText( i18n( "Delete" ) ); + removeAuthorButton->setFlat( true ); + vboxl->addWidget( removeAuthorButton ); + vboxl->addStretch(); + + layout->addLayout( vboxl ); + + //Connect Signals & Slots + + connect ( newAuthorButton, TQ_SIGNAL( clicked() ), authorListView, TQ_SLOT( createNew() ) ); + connect ( removeAuthorButton, TQ_SIGNAL( clicked() ), authorListView, TQ_SLOT( remove + () ) ); +} + +AuthorsDialog::~AuthorsDialog() +{} + +// (Re)loads the data from the database +void AuthorsDialog::reload( ReloadFlags flag ) +{ + authorListView->reload( flag ); +} + +#include "authorsdialog.moc" diff --git a/src/dialogs/authorsdialog.h b/src/dialogs/authorsdialog.h new file mode 100644 index 0000000..3678ae3 --- /dev/null +++ b/src/dialogs/authorsdialog.h @@ -0,0 +1,51 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef AUTHORSDIALOG_H +#define AUTHORSDIALOG_H + +#include <tqwidget.h> +#include <tqpushbutton.h> +#include <tqhbox.h> +#include <tqlayout.h> +#include <kiconloader.h> +#include <tdelistview.h> + +#include "widgets/dblistviewbase.h" + +class RecipeDB; +class StdAuthorListView; + +/** +@author Unai Garro +*/ + +class AuthorsDialog: public TQWidget +{ + + TQ_OBJECT + +public: + + AuthorsDialog( TQWidget* parent, RecipeDB *db ); + ~AuthorsDialog(); + void reload( ReloadFlags flag = Load ); +private: + // Internal data + RecipeDB *database; + //Widgets + StdAuthorListView *authorListView; + TQPushButton *newAuthorButton; + TQPushButton *removeAuthorButton; + TDEIconLoader *il; +}; +#endif diff --git a/src/dialogs/borderdialog.cpp b/src/dialogs/borderdialog.cpp new file mode 100644 index 0000000..13a8459 --- /dev/null +++ b/src/dialogs/borderdialog.cpp @@ -0,0 +1,261 @@ +/*************************************************************************** +* Copyright (C) 2004 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "borderdialog.h" + +#include <tqpushbutton.h> +#include <tqgroupbox.h> +#include <tqvbox.h> +#include <tqlabel.h> +#include <tqspinbox.h> +#include <tqlayout.h> +#include <tqtooltip.h> +#include <tqwhatsthis.h> + +#include <kdebug.h> +#include <tdehtml_part.h> +#include <tdehtmlview.h> +#include <tdelistbox.h> +#include <tdelocale.h> + +#include "datablocks/kreborder.h" + +BorderDialog::BorderDialog( const KreBorder &border, TQWidget* parent, const char* name ) + : KDialogBase( parent, name, true, TQString::null, + KDialogBase::Ok | KDialogBase::Cancel, KDialogBase::Ok ) +{ + TQVBox *page = makeVBoxMainWidget(); + + borderGroupBox = new TQGroupBox( page, "borderGroupBox" ); + borderGroupBox->setColumnLayout( 0, TQt::Vertical ); + borderGroupBox->layout() ->setSpacing( 6 ); + borderGroupBox->layout() ->setMargin( 11 ); + borderGroupBoxLayout = new TQVBoxLayout( borderGroupBox->layout() ); + borderGroupBoxLayout->setAlignment( TQt::AlignTop ); + + layout4 = new TQHBoxLayout( 0, 0, 6, "layout4" ); + + layout3 = new TQVBoxLayout( 0, 0, 6, "layout3" ); + + styleLabel = new TQLabel( borderGroupBox, "styleLabel" ); + layout3->addWidget( styleLabel ); + + styleListBox = new TDEListBox( borderGroupBox, "styleListBox" ); + layout3->addWidget( styleListBox ); + layout4->addLayout( layout3 ); + + layout2 = new TQVBoxLayout( 0, 0, 6, "layout2" ); + + colorLabel = new TQLabel( borderGroupBox, "colorLabel" ); + layout2->addWidget( colorLabel ); + + TQHBox *color_hbox = new TQHBox( borderGroupBox ); + color_hbox->setSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::MinimumExpanding ); + hsSelector = new KHSSelector( color_hbox ); + hsSelector->setMinimumSize( 140, 70 ); + connect( hsSelector, TQ_SIGNAL( valueChanged( int, int ) ), TQ_SLOT( slotHSChanged( int, int ) ) ); + + valuePal = new KValueSelector( color_hbox ); + valuePal->setMinimumSize( 26, 70 ); + connect( valuePal, TQ_SIGNAL( valueChanged( int ) ), TQ_SLOT( slotVChanged( int ) ) ); + + layout2->addWidget( color_hbox ); + layout4->addLayout( layout2 ); + + layout1 = new TQVBoxLayout( 0, 0, 6, "layout1" ); + + widthLabel = new TQLabel( borderGroupBox, "widthLabel" ); + layout1->addWidget( widthLabel ); + + widthSpinBox = new TQSpinBox( borderGroupBox, "widthSpinBox" ); + widthSpinBox->setMinValue( 1 ); + layout1->addWidget( widthSpinBox ); + + widthListBox = new TDEListBox( borderGroupBox, "widthListBox" ); + layout1->addWidget( widthListBox ); + layout4->addLayout( layout1 ); + borderGroupBoxLayout->addLayout( layout4 ); + + borderPreview = new TDEHTMLPart( borderGroupBox ); + borderPreview->view() ->setSizePolicy( TQSizePolicy::Preferred, TQSizePolicy::Fixed ); + borderGroupBoxLayout->addWidget( borderPreview->view() ); + + languageChange(); + + connect( widthSpinBox, TQ_SIGNAL( valueChanged( int ) ), TQ_SLOT( updatePreview() ) ); + connect( widthListBox, TQ_SIGNAL( highlighted( int ) ), TQ_SLOT( updateSpinBox( int ) ) ); + connect( styleListBox, TQ_SIGNAL( highlighted( int ) ), TQ_SLOT( updatePreview() ) ); + + initListBoxs(); + loadBorder( border ); + + clearWState( WState_Polished ); +} + +BorderDialog::~BorderDialog() +{} + +void BorderDialog::languageChange() +{ + borderGroupBox->setTitle( i18n( "Requested Border" ) ); + styleLabel->setText( i18n( "Style:" ) ); + colorLabel->setText( i18n( "Color:" ) ); + widthLabel->setText( i18n( "Width:" ) ); +} + +KreBorder BorderDialog::border() const +{ + int width = widthSpinBox->value(); + + TQString style; + switch ( styleListBox->currentItem() ) { + case 0: + style = "none"; + break; + case 1: + style = "dotted"; + break; + case 2: + style = "dashed"; + break; + case 3: + style = "solid"; + break; + case 4: + style = "double"; + break; + case 5: + style = "groove"; + break; + case 6: + style = "ridge"; + break; + case 7: + style = "inset"; + break; + case 8: + style = "outset"; + break; + } + + return KreBorder( width, style, selColor ); +} + +void BorderDialog::loadBorder( const KreBorder &border ) +{ + widthSpinBox->setValue( border.width ); + widthListBox->setCurrentItem( border.width - 1 ); + + if ( border.style == "none" ) + styleListBox->setCurrentItem( 0 ); + else if ( border.style == "dotted" ) + styleListBox->setCurrentItem( 1 ); + else if ( border.style == "dashed" ) + styleListBox->setCurrentItem( 2 ); + else if ( border.style == "solid" ) + styleListBox->setCurrentItem( 3 ); + else if ( border.style == "double" ) + styleListBox->setCurrentItem( 4 ); + else if ( border.style == "groove" ) + styleListBox->setCurrentItem( 5 ); + else if ( border.style == "ridge" ) + styleListBox->setCurrentItem( 6 ); + else if ( border.style == "inset" ) + styleListBox->setCurrentItem( 7 ); + else if ( border.style == "outset" ) + styleListBox->setCurrentItem( 8 ); + + setColor( border.color ); + + updatePreview(); +} + +void BorderDialog::initListBoxs() +{ + styleListBox->insertItem( i18n( "None" ) ); + styleListBox->insertItem( i18n( "See http://krecipes.sourceforge.net/bordertypes.png for an example", "Dotted" ) ); + styleListBox->insertItem( i18n( "See http://krecipes.sourceforge.net/bordertypes.png for an example", "Dashed" ) ); + styleListBox->insertItem( i18n( "See http://krecipes.sourceforge.net/bordertypes.png for an example", "Solid" ) ); + styleListBox->insertItem( i18n( "See http://krecipes.sourceforge.net/bordertypes.png for an example", "Double" ) ); + styleListBox->insertItem( i18n( "See http://krecipes.sourceforge.net/bordertypes.png for an example", "Groove" ) ); + styleListBox->insertItem( i18n( "See http://krecipes.sourceforge.net/bordertypes.png for an example", "Ridge" ) ); + styleListBox->insertItem( i18n( "See http://krecipes.sourceforge.net/bordertypes.png for an example", "Inset" ) ); + styleListBox->insertItem( i18n( "See http://krecipes.sourceforge.net/bordertypes.png for an example", "Outset" ) ); + + widthListBox->insertItem( "1" ); + widthListBox->insertItem( "2" ); + widthListBox->insertItem( "3" ); + widthListBox->insertItem( "4" ); + widthListBox->insertItem( "5" ); + widthListBox->insertItem( "6" ); + widthListBox->insertItem( "7" ); +} + +void BorderDialog::updatePreview() +{ + KreBorder b( border() ); + + TQString html_str = TQString( "<html><body><div style=\"vertical-align: middle; border: %1px %2 %3;\"><center><h1>%4</h1></center></div></body></html>" ).arg( b.width ).arg( b.style ).arg( b.color.name() ).arg( i18n( "Border Preview" ) ); + + borderPreview->begin(); + borderPreview->write( html_str ); + borderPreview->end(); + borderPreview->show(); +} + +void BorderDialog::updateSpinBox( int index ) +{ + widthSpinBox->setValue( index + 1 ); +} + +void BorderDialog::slotHSChanged( int h, int s ) +{ + int _h, _s, v; + selColor.hsv( &_h, &_s, &v ); + if ( v < 1 ) + v = 1; + + KColor col; + col.setHsv( h, s, v ); + + setColor( col ); + updatePreview(); +} + +void BorderDialog::slotVChanged( int v ) +{ + int h, s, _v; + selColor.hsv( &h, &s, &_v ); + + KColor col; + col.setHsv( h, s, v ); + + setColor( col ); + updatePreview(); +} + +void BorderDialog::setColor( const KColor &color ) +{ + if ( color == selColor ) + return ; + + selColor = color; + + int h, s, v; + color.hsv( &h, &s, &v ); + hsSelector->setValues( h, s ); + valuePal->setHue( h ); + valuePal->setSaturation( s ); + valuePal->setValue( v ); + valuePal->updateContents(); + valuePal->repaint( FALSE ); +} + +#include "borderdialog.moc" diff --git a/src/dialogs/borderdialog.h b/src/dialogs/borderdialog.h new file mode 100644 index 0000000..07905ee --- /dev/null +++ b/src/dialogs/borderdialog.h @@ -0,0 +1,75 @@ +/*************************************************************************** +* Copyright (C) 2004 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef BORDERDIALOG_H +#define BORDERDIALOG_H + +#include <kdialogbase.h> +#include <kcolordialog.h> + +class TQVBoxLayout; +class TQHBoxLayout; +class TQGridLayout; +class TQSpacerItem; +class TQGroupBox; +class TQLabel; +class TDEListBox; +class TQListBoxItem; +class TQSpinBox; +class TDEHTMLPart; + +class KreBorder; + +class BorderDialog : public KDialogBase +{ + TQ_OBJECT + +public: + BorderDialog( const KreBorder &border, TQWidget* parent = 0, const char* name = 0 ); + ~BorderDialog(); + + KreBorder border() const; + + TQGroupBox* borderGroupBox; + TQLabel* styleLabel; + TDEListBox* styleListBox; + TQLabel* colorLabel; + KHSSelector* hsSelector; + KValueSelector* valuePal; + TQLabel* widthLabel; + TQSpinBox* widthSpinBox; + TDEListBox* widthListBox; + TDEHTMLPart* borderPreview; + +protected: + TQVBoxLayout* borderGroupBoxLayout; + TQHBoxLayout* layout4; + TQVBoxLayout* layout3; + TQVBoxLayout* layout2; + TQVBoxLayout* layout1; + +protected slots: + virtual void languageChange(); + void updatePreview(); + void updateSpinBox( int index ); + + void slotHSChanged( int h, int s ); + void slotVChanged( int v ); + void setColor( const KColor &color ); + +private: + void loadBorder( const KreBorder &border ); + void initListBoxs(); + + KColor selColor; + +}; + +#endif // BORDERDIALOG_H diff --git a/src/dialogs/categorieseditordialog.cpp b/src/dialogs/categorieseditordialog.cpp new file mode 100644 index 0000000..d383fde --- /dev/null +++ b/src/dialogs/categorieseditordialog.cpp @@ -0,0 +1,67 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "categorieseditordialog.h" + +#include <kdebug.h> +#include <kdialog.h> +#include <tdelocale.h> +#include <tdemessagebox.h> + +#include "widgets/categorylistview.h" +#include "createcategorydialog.h" +#include "backends/recipedb.h" + +CategoriesEditorDialog::CategoriesEditorDialog( TQWidget* parent, RecipeDB *db ) : TQWidget( parent ) +{ + + // Store pointer to database + database = db; + + TQHBoxLayout* layout = new TQHBoxLayout( this, KDialog::marginHint(), KDialog::spacingHint() ); + + //Category List + categoryListView = new StdCategoryListView( this, database, true ); + layout->addWidget( categoryListView ); + + //Buttons + TQVBoxLayout* vboxl = new TQVBoxLayout( KDialog::spacingHint() ); + + newCategoryButton = new TQPushButton( this ); + newCategoryButton->setText( i18n( "Create ..." ) ); + newCategoryButton->setFlat( true ); + vboxl->addWidget( newCategoryButton ); + + removeCategoryButton = new TQPushButton( this ); + removeCategoryButton->setText( i18n( "Delete" ) ); + removeCategoryButton->setFlat( true ); + vboxl->addWidget( removeCategoryButton ); + vboxl->addStretch(); + + layout->addLayout( vboxl ); + + //Connect Signals & Slots + + connect ( newCategoryButton, TQ_SIGNAL( clicked() ), categoryListView, TQ_SLOT( createNew() ) ); + connect ( removeCategoryButton, TQ_SIGNAL( clicked() ), categoryListView, TQ_SLOT( remove + () ) ); +} + +CategoriesEditorDialog::~CategoriesEditorDialog() +{} + +void CategoriesEditorDialog::reload( ReloadFlags flag ) +{ + categoryListView->reload( flag ); +} + +#include "categorieseditordialog.moc" diff --git a/src/dialogs/categorieseditordialog.h b/src/dialogs/categorieseditordialog.h new file mode 100644 index 0000000..9b2e56d --- /dev/null +++ b/src/dialogs/categorieseditordialog.h @@ -0,0 +1,56 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef CATEGORIESEDITORDIALOG_H +#define CATEGORIESEDITORDIALOG_H + +#include <tqwidget.h> +#include <tqpushbutton.h> +#include <tqhbox.h> +#include <tqlayout.h> +#include <kiconloader.h> +#include <tdelistview.h> + +#include "datablocks/categorytree.h" +#include "widgets/dblistviewbase.h" + +class RecipeDB; +class StdCategoryListView; + +/** +@author Unai Garro +*/ +class CategoriesEditorDialog: public TQWidget +{ + + TQ_OBJECT + +public: + + CategoriesEditorDialog( TQWidget* parent, RecipeDB *db ); + ~CategoriesEditorDialog(); + + void reload( ReloadFlags flag = Load ); + +private: + // Internal data + RecipeDB *database; + //Widgets + TQGridLayout *layout; + StdCategoryListView *categoryListView; + TQHBox *buttonBar; + TQPushButton *newCategoryButton; + TQPushButton *removeCategoryButton; + TDEIconLoader *il; +}; + +#endif diff --git a/src/dialogs/conversiondialog.cpp b/src/dialogs/conversiondialog.cpp new file mode 100644 index 0000000..c9e3f51 --- /dev/null +++ b/src/dialogs/conversiondialog.cpp @@ -0,0 +1,166 @@ +/*************************************************************************** +* Copyright (C) 2006 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "conversiondialog.h" + +#include <tqvariant.h> +#include <tqpushbutton.h> +#include <kcombobox.h> +#include <tqlabel.h> +#include <klineedit.h> +#include <tqlayout.h> +#include <tqtooltip.h> +#include <tqwhatsthis.h> +#include <tqvbox.h> + +#include <kcombobox.h> +#include <klineedit.h> +#include <tdelocale.h> + +#include "backends/recipedb.h" +#include "widgets/unitcombobox.h" +#include "widgets/ingredientcombobox.h" +#include "widgets/prepmethodcombobox.h" +#include "widgets/fractioninput.h" + +ConversionDialog::ConversionDialog( TQWidget* parent, RecipeDB *db, const char* name ) + : KDialogBase( parent, name, false, i18n( "Measurement Converter" ), + KDialogBase::Close | KDialogBase::User1 | KDialogBase::Help, KDialogBase::Close ), + m_database(db) +{ + setHelp("measure-converter"); + setButtonText( KDialogBase::User1, i18n("Convert") ); + + setSizeGripEnabled( TRUE ); + + TQVBox *page = makeVBoxMainWidget(); + + TQHBox *vbox = new TQVBox(page); + + TQHBox *fromTopBox = new TQHBox(vbox); + convertLabel = new TQLabel( fromTopBox, "convertLabel" ); + + amountEdit = new FractionInput( fromTopBox ); + + fromUnitBox = new UnitComboBox( fromTopBox, db ); + fromUnitBox->reload(); + fromTopBox->setStretchFactor( fromUnitBox, 2 ); + fromTopBox->setSpacing(3); + + TQHBox *fromBottomBox = new TQHBox(vbox); + + ingredientBox = new IngredientComboBox( FALSE, fromBottomBox, db, i18n( "--Ingredient (optional)--" ) ); + ingredientBox->reload(); + + prepMethodBox = new PrepMethodComboBox( false, fromBottomBox, db, i18n( "-No Preparation-" ) ); + prepMethodBox->reload(); + fromBottomBox->setSpacing(3); + + TQHBox *toBox = new TQHBox(vbox); + + toLabel = new TQLabel( toBox, "toLabel" ); + + toUnitBox = new UnitComboBox( toBox, db ); + toUnitBox->reload(); + toBox->setStretchFactor( toUnitBox, 2 ); + toBox->setSpacing(8); + + TQHBox *resultBox = new TQHBox(vbox); + resultLabel = new TQLabel( resultBox, "resultLabel" ); + resultText = new TQLabel( resultBox, "resultText" ); + resultBox->setStretchFactor( resultText, 2 ); + + languageChange(); + + setInitialSize( TQSize(300, 200).expandedTo(minimumSizeHint()) ); + + // signals and slots connections + connect ( this, TQ_SIGNAL( closeClicked() ), this, TQ_SLOT( accept() ) ); +} + +ConversionDialog::~ConversionDialog() +{ +} + +void ConversionDialog::languageChange() +{ + convertLabel->setText( i18n( "Convert" ) ); + toLabel->setText( i18n( "To" ) ); + resultLabel->setText( i18n( "<b>Result:</b>" ) ); + resultText->setText( TQString::null ); +} + +void ConversionDialog::show() +{ + reset(); + KDialogBase::show(); +} + +void ConversionDialog::reset() +{ + resultText->setText( TQString::null ); + ingredientBox->setCurrentItem( 0 ); + prepMethodBox->setCurrentItem( 0 ); + toUnitBox->setCurrentItem( 0 ); + fromUnitBox->setCurrentItem( 0 ); + amountEdit->clear(); +} + +void ConversionDialog::slotUser1() +{ + convert(); +} + +void ConversionDialog::convert() +{ + Ingredient result, ing; + Unit unit = m_database->unitName(toUnitBox->id(toUnitBox->currentItem())); + + ing.amount = amountEdit->value().toDouble(); + ing.ingredientID = ingredientBox->id(ingredientBox->currentItem()); + ing.units = m_database->unitName(fromUnitBox->id(fromUnitBox->currentItem())); + + int prepID = prepMethodBox->id(prepMethodBox->currentItem()); + if ( prepID != -1 ) + ing.prepMethodList.append(Element(TQString::null,prepID)); + + switch ( m_database->convertIngredientUnits( ing, unit, result ) ) { + case RecipeDB::Success: + resultLabel->setText( i18n( "<b>Result:</b>" ) ); + resultText->setText(TQString::number(result.amount)+" "+((result.amount>1)?result.units.plural:result.units.name)); + break; + case RecipeDB::MismatchedPrepMethodUsingApprox: + resultLabel->setText( i18n( "<b>Approximated result:</b>" ) ); + resultText->setText(TQString::number(result.amount)+" "+((result.amount>1)?result.units.plural:result.units.name)); + break; + case RecipeDB::MissingUnitConversion: + resultLabel->setText( i18n( "<b>Error:</b>" ) ); + resultText->setText( i18n("Missing unit conversion") ); + break; + case RecipeDB::MissingIngredientWeight: + resultLabel->setText( i18n( "<b>Error:</b>" ) ); + resultText->setText( i18n("No ingredient weight available") ); + break; + case RecipeDB::MismatchedPrepMethod: + resultLabel->setText( i18n( "<b>Error:</b>" ) ); + resultText->setText( i18n("No ingredient weight available for this method of preparation") ); + break; + case RecipeDB::MissingIngredient: + resultLabel->setText( i18n( "<b>Error:</b>" ) ); + resultText->setText( i18n("Ingredient required for conversion") ); + break; + case RecipeDB::InvalidTypes: + resultLabel->setText( i18n( "<b>Error:</b>" ) ); + resultText->setText( i18n("Impossible unit conversion based on unit types") ); + break; + } +} + +#include "conversiondialog.moc" diff --git a/src/dialogs/conversiondialog.h b/src/dialogs/conversiondialog.h new file mode 100644 index 0000000..b467ce1 --- /dev/null +++ b/src/dialogs/conversiondialog.h @@ -0,0 +1,62 @@ +/*************************************************************************** +* Copyright (C) 2006 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef CONVERSIONDIALOG_H +#define CONVERSIONDIALOG_H + +#include <kdialogbase.h> + +class TQVBoxLayout; +class TQHBoxLayout; +class TQGridLayout; +class TQSpacerItem; +class KComboBox; +class TQLabel; +class KLineEdit; +class TQPushButton; + +class RecipeDB; +class UnitComboBox; +class IngredientComboBox; +class PrepMethodComboBox; +class FractionInput; + +class ConversionDialog : public KDialogBase +{ +TQ_OBJECT + +public: + ConversionDialog( TQWidget* parent, RecipeDB *, const char* name = 0 ); + ~ConversionDialog(); + + virtual void show(); + void reset(); + +protected: + IngredientComboBox* ingredientBox; + PrepMethodComboBox* prepMethodBox; + TQLabel* convertLabel; + UnitComboBox* toUnitBox; + UnitComboBox* fromUnitBox; + FractionInput* amountEdit; + TQLabel* toLabel; + TQLabel* resultLabel; + TQLabel* resultText; + +protected slots: + virtual void languageChange(); + void slotUser1(); + void convert(); + +private: + RecipeDB *m_database; +}; + +#endif // CONVERSIONDIALOG_H diff --git a/src/dialogs/createcategorydialog.cpp b/src/dialogs/createcategorydialog.cpp new file mode 100644 index 0000000..f0cbc21 --- /dev/null +++ b/src/dialogs/createcategorydialog.cpp @@ -0,0 +1,80 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "createcategorydialog.h" + +#include <tqpushbutton.h> +#include <tqgroupbox.h> +#include <tqlayout.h> +#include <tqlabel.h> +#include <tqvbox.h> + +#include <kcombobox.h> +#include <klineedit.h> +#include <tdelocale.h> + +CreateCategoryDialog::CreateCategoryDialog( TQWidget *parent, const ElementList& categories ) + : KDialogBase( parent, "createCategoryDialog", true, i18n( "New Category" ), + KDialogBase::Ok | KDialogBase::Cancel, KDialogBase::Ok ) +{ + TQVBox *page = makeVBoxMainWidget(); + + box = new TQGroupBox( page ); + box->setColumnLayout( 0, TQt::Vertical ); + box->layout() ->setSpacing( 6 ); + box->layout() ->setMargin( 11 ); + TQVBoxLayout *boxLayout = new TQVBoxLayout( box->layout() ); + boxLayout->setAlignment( TQt::AlignTop ); + box->setTitle( i18n( "New Category" ) ); + + elementEdit = new KLineEdit( box ); + boxLayout->addWidget( elementEdit ); + + TQHBox *subcatHBox = new TQHBox( box ); + ( void ) new TQLabel( i18n( "Subcategory of:" ), subcatHBox ); + categoryComboBox = new KComboBox( subcatHBox ); + boxLayout->addWidget( subcatHBox ); + loadCategories( categories ); + + adjustSize(); + setFixedSize( size() ); //we've got all the widgets put in, now let's keep it this size + + elementEdit->setFocus(); +} + + +CreateCategoryDialog::~CreateCategoryDialog() +{} + +void CreateCategoryDialog::loadCategories( const ElementList& categories ) +{ + categoryComboBox->insertItem( i18n( "**NONE**" ) ); + for ( ElementList::const_iterator it = categories.begin(); it != categories.end(); ++it ) { + categoryComboBox->insertItem( ( *it ).name ); + idMap.insert( ( *it ).name, ( *it ).id ); + } +} + +TQString CreateCategoryDialog::newCategoryName( void ) +{ + return ( elementEdit->text() ); +} + +int CreateCategoryDialog::subcategory( void ) +{ + if ( categoryComboBox->currentItem() == 0 ) { + return -1; + } + else { + return idMap[ categoryComboBox->currentText() ]; + } +} diff --git a/src/dialogs/createcategorydialog.h b/src/dialogs/createcategorydialog.h new file mode 100644 index 0000000..2211006 --- /dev/null +++ b/src/dialogs/createcategorydialog.h @@ -0,0 +1,51 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef CREATECATEGORYDIALOG_H +#define CREATECATEGORYDIALOG_H + +#include <tqmap.h> + +#include <kdialogbase.h> + +#include "datablocks/elementlist.h" + +class KLineEdit; +class TQPushButton; +class TQVBoxLayout; +class TQGroupBox; +class TQVBox; +class KComboBox; + +/** +@author Jason Kivlighn +*/ +class CreateCategoryDialog : public KDialogBase +{ +public: + CreateCategoryDialog( TQWidget *parent, const ElementList &categories ); + ~CreateCategoryDialog(); + TQString newCategoryName( void ); + int subcategory( void ); + +private: + void loadCategories( const ElementList &categories ); + + //Widgets + TQGroupBox *box; + KLineEdit *elementEdit; + KComboBox* categoryComboBox; + TQMap<TQString, int> idMap; + +}; + +#endif diff --git a/src/dialogs/createelementdialog.cpp b/src/dialogs/createelementdialog.cpp new file mode 100644 index 0000000..1f5b4a0 --- /dev/null +++ b/src/dialogs/createelementdialog.cpp @@ -0,0 +1,48 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "createelementdialog.h" + +#include <tdelocale.h> + +CreateElementDialog::CreateElementDialog( TQWidget *parent, const TQString &text ) + : KDialogBase( parent, "createElementDialog", true, text, + KDialogBase::Ok | KDialogBase::Cancel, KDialogBase::Ok ) +{ + TQVBox *page = makeVBoxMainWidget(); + + box = new TQGroupBox( page ); + box->setColumnLayout( 0, TQt::Vertical ); + box->layout() ->setSpacing( 6 ); + box->layout() ->setMargin( 11 ); + TQVBoxLayout *boxLayout = new TQVBoxLayout( box->layout() ); + boxLayout->setAlignment( TQt::AlignTop ); + box->setTitle( text ); + + elementEdit = new KLineEdit( box ); + boxLayout->addWidget( elementEdit ); + + adjustSize(); + setFixedSize( size() ); //we've got all the widgets put in, now let's keep it this size + + elementEdit->setFocus(); +} + + +CreateElementDialog::~CreateElementDialog() +{} + +TQString CreateElementDialog::newElementName( void ) +{ + return ( elementEdit->text() ); +} + diff --git a/src/dialogs/createelementdialog.h b/src/dialogs/createelementdialog.h new file mode 100644 index 0000000..775647a --- /dev/null +++ b/src/dialogs/createelementdialog.h @@ -0,0 +1,40 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef CREATEELEMENTDIALOG_H +#define CREATEELEMENTDIALOG_H + +#include <tqpushbutton.h> +#include <tqgroupbox.h> +#include <tqlayout.h> +#include <tqvbox.h> + +#include <klineedit.h> +#include <kdialogbase.h> + +/** +@author Unai Garro +*/ +class CreateElementDialog : public KDialogBase +{ +public: + CreateElementDialog( TQWidget *parent, const TQString &text ); + ~CreateElementDialog(); + TQString newElementName( void ); + +private: + //Widgets + TQGroupBox *box; + KLineEdit *elementEdit; +}; + +#endif diff --git a/src/dialogs/createingredientweightdialog.cpp b/src/dialogs/createingredientweightdialog.cpp new file mode 100644 index 0000000..45bf158 --- /dev/null +++ b/src/dialogs/createingredientweightdialog.cpp @@ -0,0 +1,124 @@ +/*************************************************************************** +* Copyright (C) 2006 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "createingredientweightdialog.h" + +#include <tqgroupbox.h> +#include <tqlabel.h> +#include <tqpushbutton.h> +#include <tqlayout.h> +#include <tqtooltip.h> +#include <tqwhatsthis.h> +#include <tqvbox.h> + +#include <tdemessagebox.h> +#include <tdelocale.h> + +#include "widgets/unitcombobox.h" +#include "widgets/prepmethodcombobox.h" +#include "widgets/fractioninput.h" +#include "datablocks/weight.h" +#include "backends/recipedb.h" + +CreateIngredientWeightDialog::CreateIngredientWeightDialog( TQWidget* parent, RecipeDB *db ) + : KDialogBase( parent, "createIngWeightDialog", true, TQString::null, + KDialogBase::Ok | KDialogBase::Cancel, KDialogBase::Ok ) +{ + TQVBox *page = makeVBoxMainWidget(); + + groupBox1 = new TQGroupBox( page ); + groupBox1->setColumnLayout(0, TQt::Vertical ); + groupBox1->layout()->setSpacing( 6 ); + groupBox1->layout()->setMargin( 11 ); + groupBox1Layout = new TQGridLayout( groupBox1->layout() ); + groupBox1Layout->setAlignment( TQt::AlignTop ); + + perAmountEdit = new FractionInput( groupBox1 ); + + groupBox1Layout->addWidget( perAmountEdit, 1, 1 ); + + weightEdit = new FractionInput( groupBox1 ); + + groupBox1Layout->addWidget( weightEdit, 0, 1 ); + + weightUnitBox = new UnitComboBox( groupBox1, db, Unit::Mass ); + weightUnitBox->reload(); + + groupBox1Layout->addMultiCellWidget( weightUnitBox, 0, 0, 2, 3 ); + + perAmountLabel = new TQLabel( groupBox1, "perAmountLabel" ); + + groupBox1Layout->addWidget( perAmountLabel, 1, 0 ); + + weightLabel = new TQLabel( groupBox1, "weightLabel" ); + + groupBox1Layout->addWidget( weightLabel, 0, 0 ); + + perAmountUnitBox = new UnitComboBox( groupBox1, db ); + perAmountUnitBox->reload(); + + groupBox1Layout->addWidget( perAmountUnitBox, 1, 2 ); + + prepMethodBox = new PrepMethodComboBox( false, groupBox1, db, i18n("-No Preparation-") ); + prepMethodBox->reload(); + groupBox1Layout->addWidget( prepMethodBox, 1, 3 ); + + languageChange(); + clearWState( WState_Polished ); + + weightEdit->setFocus(); +} + +CreateIngredientWeightDialog::~CreateIngredientWeightDialog() +{ + // no need to delete child widgets, TQt does it all for us +} + +void CreateIngredientWeightDialog::languageChange() +{ + groupBox1->setTitle( i18n( "New Ingredient Weight" ) ); + perAmountLabel->setText( i18n( "Per Amount:" ) ); + weightLabel->setText( i18n( "Weight:" ) ); +} + +void CreateIngredientWeightDialog::slotOk() +{ + if ( !perAmountEdit->isInputValid() ) { + KMessageBox::error( this, i18n( "Amount field contains invalid input." ), + i18n( "Invalid input" ) ); + perAmountEdit->setFocus(); + perAmountEdit->selectAll(); + return; + } + else if ( !weightEdit->isInputValid() ) { + KMessageBox::error( this, i18n( "Amount field contains invalid input." ), + i18n( "Invalid input" ) ); + weightEdit->setFocus(); + weightEdit->selectAll(); + return; + } + + accept(); +} + +Weight CreateIngredientWeightDialog::weight() const +{ + Weight w; + w.perAmount = perAmountEdit->value().toDouble(); + w.perAmountUnitID = perAmountUnitBox->id( perAmountUnitBox->currentItem() ); + w.weight = weightEdit->value().toDouble(); + w.weightUnitID = weightUnitBox->id( weightUnitBox->currentItem() ); + w.prepMethodID = prepMethodBox->id( prepMethodBox->currentItem() ); + w.prepMethod = prepMethodBox->currentText(); + + return w; +} + +#include "createingredientweightdialog.moc" diff --git a/src/dialogs/createingredientweightdialog.h b/src/dialogs/createingredientweightdialog.h new file mode 100644 index 0000000..cbc4d8f --- /dev/null +++ b/src/dialogs/createingredientweightdialog.h @@ -0,0 +1,60 @@ +/*************************************************************************** +* Copyright (C) 2006 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef CREATEINGREDIENTWEIGHTDIALOG_H +#define CREATEINGREDIENTWEIGHTDIALOG_H + +#include <kdialogbase.h> + +class TQVBoxLayout; +class TQHBoxLayout; +class TQGridLayout; +class TQSpacerItem; +class TQGroupBox; +class KLineEdit; +class KComboBox; +class TQLabel; +class TQPushButton; + +class FractionInput; +class UnitComboBox; +class PrepMethodComboBox; +class RecipeDB; +class Weight; + +class CreateIngredientWeightDialog : public KDialogBase +{ +TQ_OBJECT + +public: + CreateIngredientWeightDialog( TQWidget* parent, RecipeDB* ); + ~CreateIngredientWeightDialog(); + + Weight weight() const; + +protected: + TQGridLayout* groupBox1Layout; + +protected slots: + virtual void languageChange(); + void slotOk(); + +private: + TQGroupBox* groupBox1; + FractionInput* perAmountEdit; + FractionInput* weightEdit; + UnitComboBox* weightUnitBox; + TQLabel* perAmountLabel; + TQLabel* weightLabel; + UnitComboBox* perAmountUnitBox; + PrepMethodComboBox* prepMethodBox; +}; + +#endif // CREATEINGREDIENTWEIGHTDIALOG_H diff --git a/src/dialogs/createpropertydialog.cpp b/src/dialogs/createpropertydialog.cpp new file mode 100644 index 0000000..9f8f107 --- /dev/null +++ b/src/dialogs/createpropertydialog.cpp @@ -0,0 +1,69 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "createpropertydialog.h" + +#include <tdelocale.h> + +CreatePropertyDialog::CreatePropertyDialog( TQWidget *parent, UnitList* list ) + : KDialogBase( parent, "createPropertyDialog", true, i18n( "New Property" ), + KDialogBase::Ok | KDialogBase::Cancel, KDialogBase::Ok ) +{ + + // Initialize Internal Variables + unitList = list; // Store the pointer to the unitList; + + // Initialize widgets + TQVBox *page = makeVBoxMainWidget(); + + box = new TQGroupBox( page ); + box->setColumnLayout( 0, TQt::Vertical ); + box->layout() ->setSpacing( 6 ); + box->layout() ->setMargin( 11 ); + TQGridLayout *gridLayout = new TQGridLayout( box->layout() ); + gridLayout->setAlignment( TQt::AlignTop ); + box->setTitle( i18n( "New Property" ) ); + + nameEditText = new TQLabel( i18n( "Property name:" ), box ); + propertyNameEdit = new KLineEdit( box ); + propertyNameEdit->setMinimumWidth( 150 ); + gridLayout->addWidget( nameEditText, 0, 0 ); + gridLayout->addWidget( propertyNameEdit, 0, 1 ); + + unitsText = new TQLabel( i18n( "Units:" ), box ); + propertyUnits = new KLineEdit( box ); + propertyUnits->setMinimumWidth( 150 ); + gridLayout->addWidget( unitsText, 1, 0 ); + gridLayout->addWidget( propertyUnits, 1, 1 ); + + adjustSize(); + setFixedSize( size() ); + + propertyNameEdit->setFocus(); +} + + +CreatePropertyDialog::~CreatePropertyDialog() +{} + + +TQString CreatePropertyDialog::newPropertyName( void ) +{ + return ( propertyNameEdit->text() ); +} + +TQString CreatePropertyDialog::newUnitsName( void ) +{ + return ( propertyUnits->text() ); +} + + diff --git a/src/dialogs/createpropertydialog.h b/src/dialogs/createpropertydialog.h new file mode 100644 index 0000000..1714e9e --- /dev/null +++ b/src/dialogs/createpropertydialog.h @@ -0,0 +1,56 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +/** +@author Unai Garro +*/ + +#ifndef CREATEPROPERTYDIALOG_H +#define CREATEPROPERTYDIALOG_H + +#include <tqcombobox.h> +#include <tqlayout.h> +#include <tqpushbutton.h> +#include <tqgroupbox.h> +#include <tqvbox.h> +#include <tqlabel.h> + +#include <klineedit.h> +#include <kdialogbase.h> + +#include "datablocks/unit.h" + +class CreatePropertyDialog : public KDialogBase +{ +public: + CreatePropertyDialog( TQWidget *parent, UnitList *list ); + ~CreatePropertyDialog(); + TQString newPropertyName( void ); + TQString newUnitsName( void ); + + +private: + //Widgets + TQGroupBox *box; + KLineEdit *propertyNameEdit; + KLineEdit *propertyUnits; + TQLabel *nameEditText; + TQLabel *unitsText; + + //Internal variables + UnitList *unitList; + + //Methods + void loadUnits(); +}; + +#endif diff --git a/src/dialogs/createunitdialog.cpp b/src/dialogs/createunitdialog.cpp new file mode 100644 index 0000000..4929bb6 --- /dev/null +++ b/src/dialogs/createunitdialog.cpp @@ -0,0 +1,116 @@ +/*************************************************************************** +* Copyright (C) 2003-2004 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "createunitdialog.h" + +#include <tqlabel.h> + +#include <tdelocale.h> +#include <klineedit.h> +#include <kcombobox.h> + +CreateUnitDialog::CreateUnitDialog( TQWidget *parent, const TQString &name, const TQString &plural, const TQString &name_abbrev, const TQString &plural_abbrev, bool newUnit ) + : KDialogBase( parent, "createElementDialog", true, (newUnit)?i18n( "New Unit" ):i18n("Unit"), + KDialogBase::Ok | KDialogBase::Cancel, KDialogBase::Ok ) +{ + TQVBox *page = makeVBoxMainWidget(); + + box = new TQGroupBox( page ); + box->setColumnLayout( 0, TQt::Vertical ); + box->layout() ->setSpacing( 6 ); + box->layout() ->setMargin( 11 ); + TQGridLayout *gridLayout = new TQGridLayout( box->layout() ); + gridLayout->setAlignment( TQt::AlignTop ); + + box->setTitle( (newUnit)?i18n( "New Unit" ):i18n("Unit") ); + + TQLabel *nameLabel = new TQLabel( i18n( "Singular:" ), box ); + nameEdit = new KLineEdit( name, box ); + + gridLayout->addWidget( nameLabel, 0, 0 ); + gridLayout->addWidget( nameEdit, 0, 1 ); + + TQLabel *nameAbbrevLabel = new TQLabel( i18n( "Abbreviation:" ), box ); + nameAbbrevEdit = new KLineEdit( name_abbrev, box ); + + gridLayout->addWidget( nameAbbrevLabel, 0, 2 ); + gridLayout->addWidget( nameAbbrevEdit, 0, 3 ); + + TQLabel *pluralLabel = new TQLabel( i18n( "Plural:" ), box ); + pluralEdit = new KLineEdit( plural, box ); + + gridLayout->addWidget( pluralLabel, 1, 0 ); + gridLayout->addWidget( pluralEdit, 1, 1 ); + + TQLabel *pluralAbbrevLabel = new TQLabel( i18n( "Abbreviation:" ), box ); + pluralAbbrevEdit = new KLineEdit( plural_abbrev, box ); + + gridLayout->addWidget( pluralAbbrevLabel, 1, 2 ); + gridLayout->addWidget( pluralAbbrevEdit, 1, 3 ); + + TQLabel *typeLabel = new TQLabel( i18n( "Type:" ), box ); + typeComboBox = new KComboBox( false, box ); + typeComboBox->insertItem(i18n("Other")); + typeComboBox->insertItem(i18n("Mass")); + typeComboBox->insertItem(i18n("Volume")); + + gridLayout->addWidget( typeLabel, 2, 0 ); + gridLayout->addMultiCellWidget( typeComboBox, 2, 2, 1, 3 ); + + adjustSize(); + setFixedSize( size() ); //we've got all the widgets put in, now let's keep it this size + + connect( nameAbbrevEdit, TQ_SIGNAL(textChanged(const TQString&)), TQ_SLOT(nameAbbrevTextChanged(const TQString &)) ); + + if ( name.isEmpty() ) + nameEdit->setFocus(); + else if ( plural.isEmpty() ) + pluralEdit->setFocus(); +} + + +CreateUnitDialog::~CreateUnitDialog() +{} + +Unit CreateUnitDialog::newUnit( void ) +{ + TQString name = nameEdit->text(); + TQString plural = pluralEdit->text(); + + if ( name.isEmpty() ) + name = plural; + if ( plural.isEmpty() ) + plural = name; + + Unit new_unit = Unit( name, plural ); + new_unit.name_abbrev = nameAbbrevEdit->text(); + new_unit.plural_abbrev = pluralAbbrevEdit->text(); + + new_unit.type = (Unit::Type)typeComboBox->currentItem(); + + return new_unit; +} + +void CreateUnitDialog::nameAbbrevTextChanged(const TQString &newText) +{ + //appending + if ( newText.left( newText.length()-1 ) == pluralAbbrevEdit->text() ) { + pluralAbbrevEdit->setText( newText ); + } + + //truncating + if ( newText.left( newText.length()-1 ) == pluralAbbrevEdit->text().left( newText.length()-1 ) ) { + pluralAbbrevEdit->setText( newText ); + } +} + +#include "createunitdialog.moc" diff --git a/src/dialogs/createunitdialog.h b/src/dialogs/createunitdialog.h new file mode 100644 index 0000000..a2e1777 --- /dev/null +++ b/src/dialogs/createunitdialog.h @@ -0,0 +1,52 @@ +/*************************************************************************** +* Copyright (C) 2003-2004 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef CREATEUNITDIALOG_H +#define CREATEUNITDIALOG_H + +#include <kdialogbase.h> +#include <tqpushbutton.h> +#include <tqgroupbox.h> +#include <tqlayout.h> +#include <tqvbox.h> + +#include "datablocks/unit.h" + +class KComboBox; +class KLineEdit; + +/** +@author Unai Garro +*/ +class CreateUnitDialog : public KDialogBase +{ +TQ_OBJECT + +public: + CreateUnitDialog( TQWidget *parent, const TQString &name = TQString::null, const TQString &plural = TQString::null, const TQString &name_abbrev = TQString::null, const TQString &plural_abbrev = TQString::null, bool newUnit = true ); + ~CreateUnitDialog(); + Unit newUnit( void ); + +protected slots: + void nameAbbrevTextChanged(const TQString &); + +private: + //Widgets + TQGroupBox *box; + KLineEdit *nameEdit; + KLineEdit *pluralEdit; + KLineEdit *nameAbbrevEdit; + KLineEdit *pluralAbbrevEdit; + KComboBox *typeComboBox; +}; + +#endif diff --git a/src/dialogs/dbimportdialog.cpp b/src/dialogs/dbimportdialog.cpp new file mode 100644 index 0000000..7d5d68b --- /dev/null +++ b/src/dialogs/dbimportdialog.cpp @@ -0,0 +1,219 @@ +/*************************************************************************** +* Copyright (C) 2004 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "dbimportdialog.h" + +#include <unistd.h> //for getuid() +#include <pwd.h> //getpwuid() + +#ifdef HAVE_CONFIG_H +#include "config.h" +#endif + +#include <tqpushbutton.h> +#include <tqbuttongroup.h> +#include <tqradiobutton.h> +#include <tqwidgetstack.h> +#include <tqwidget.h> +#include <tqlineedit.h> +#include <tqlabel.h> +#include <tqlayout.h> +#include <tqtooltip.h> +#include <tqwhatsthis.h> +#include <tqhbox.h> + +#include <tdeconfig.h> +#include <kdebug.h> +#include <tdelocale.h> +#include <kurlrequester.h> +#include <knuminput.h> + +DBImportDialog::DBImportDialog( TQWidget *parent, const char *name ) + : KDialogBase( parent, name, true, i18n( "Database Import" ), + KDialogBase::Ok | KDialogBase::Cancel, KDialogBase::Ok ) +{ + setButtonBoxOrientation( Vertical ); + + TQHBox *page = makeHBoxMainWidget(); + + dbButtonGroup = new TQButtonGroup( page, "dbButtonGroup" ); + dbButtonGroup->setSizePolicy( TQSizePolicy( ( TQSizePolicy::SizeType ) 4, ( TQSizePolicy::SizeType ) 5, 0, 0, dbButtonGroup->sizePolicy().hasHeightForWidth() ) ); + dbButtonGroup->setColumnLayout( 0, TQt::Vertical ); + dbButtonGroup->layout() ->setSpacing( 6 ); + dbButtonGroup->layout() ->setMargin( 11 ); + dbButtonGroupLayout = new TQVBoxLayout( dbButtonGroup->layout() ); + dbButtonGroupLayout->setAlignment( TQt::AlignTop ); + + liteRadioButton = new TQRadioButton( dbButtonGroup, "liteRadioButton" ); + liteRadioButton->setChecked( TRUE ); + dbButtonGroupLayout->addWidget( liteRadioButton ); + + mysqlRadioButton = new TQRadioButton( dbButtonGroup, "mysqlRadioButton" ); + dbButtonGroupLayout->addWidget( mysqlRadioButton ); + + psqlRadioButton = new TQRadioButton( dbButtonGroup, "psqlRadioButton" ); + dbButtonGroupLayout->addWidget( psqlRadioButton ); + + paramStack = new TQWidgetStack( page, "paramStack" ); + paramStack->setSizePolicy( TQSizePolicy( ( TQSizePolicy::SizeType ) 7, ( TQSizePolicy::SizeType ) 5, 0, 0, paramStack->sizePolicy().hasHeightForWidth() ) ); + + sqlitePage = new TQWidget( paramStack, "sqlitePage" ); + serverPageLayout_2 = new TQVBoxLayout( sqlitePage, 11, 6, "serverPageLayout_2" ); + + TQLabel *sqliteLabel = new TQLabel( i18n( "Database file:" ), sqlitePage ); + serverPageLayout_2->addWidget( sqliteLabel ); + sqliteDBRequester = new KURLRequester( sqlitePage, "sqliteDBRequester" ); + sqliteDBRequester->setShowLocalProtocol( false ); + serverPageLayout_2->addWidget( sqliteDBRequester ); + + TQSpacerItem *vSpacer = new TQSpacerItem( 20, 20, TQSizePolicy::Minimum, TQSizePolicy::Expanding ); + serverPageLayout_2->addItem(vSpacer); + + paramStack->addWidget( sqlitePage, 1 ); + + serverPage = new TQWidget( paramStack, "serverPage" ); + serverPageLayout = new TQVBoxLayout( serverPage, 11, 6, "serverPageLayout" ); + + layout5 = new TQGridLayout( 0, 1, 1, 0, 6, "layout5" ); + + hostEdit = new TQLineEdit( serverPage, "hostEdit" ); + layout5->addWidget( hostEdit, 0, 1 ); + hostLabel = new TQLabel( serverPage, "hostLabel" ); + layout5->addWidget( hostLabel, 0, 0 ); + + userEdit = new TQLineEdit( serverPage, "userEdit" ); + layout5->addWidget( userEdit, 1, 1 ); + userLabel = new TQLabel( serverPage, "userLabel" ); + layout5->addWidget( userLabel, 1, 0 ); + + passwordEdit = new TQLineEdit( serverPage, "passwordEdit" ); + passwordEdit->setEchoMode( TQLineEdit::Password ); + layout5->addWidget( passwordEdit, 2, 1 ); + passwordLabel = new TQLabel( serverPage, "passwordLabel" ); + layout5->addWidget( passwordLabel, 2, 0 ); + + portEdit = new KIntNumInput( serverPage, "portEdit" ); + portEdit->setMinValue(0); + portEdit->setValue(0); + layout5->addWidget( portEdit, 3, 1 ); + portLabel = new TQLabel( serverPage, "portLabel" ); + layout5->addWidget( portLabel, 3, 0 ); + + nameEdit = new TQLineEdit( serverPage, "nameEdit" ); + layout5->addWidget( nameEdit, 4, 1 ); + nameLabel = new TQLabel( serverPage, "nameLabel" ); + layout5->addWidget( nameLabel, 4, 0 ); + + serverPageLayout->addLayout( layout5 ); + paramStack->addWidget( serverPage, 0 ); + + languageChange(); + + +#if (!HAVE_MYSQL) + + mysqlRadioButton->setEnabled( false ); +#endif + +#if (!HAVE_POSTGRESQL) + + psqlRadioButton->setEnabled( false ); +#endif + +#if (!(HAVE_SQLITE || HAVE_SQLITE3)) + + liteRadioButton->setEnabled( false ); +#if (HAVE_MYSQL) + + dbButtonGroup->setButton( 1 ); // Otherwise by default liteCheckBox is checked even if it's disabled + switchDBPage(1); +#else + #if (HAVE_POSTGRESQL) + + dbButtonGroup->setButton( 2 ); + switchDBPage(2); +#endif + #endif + #endif + + // signals and slots connections + connect( dbButtonGroup, TQ_SIGNAL( clicked( int ) ), this, TQ_SLOT( switchDBPage( int ) ) ); +} + +void DBImportDialog::languageChange() +{ + dbButtonGroup->setTitle( i18n( "Database" ) ); + liteRadioButton->setText( "SQLite" ); + mysqlRadioButton->setText( "MySQL" ); + psqlRadioButton->setText( "PostgreSQL" ); + hostLabel->setText( i18n( "Server:" ) ); + userLabel->setText( i18n( "Username:" ) ); + passwordLabel->setText( i18n( "Password:" ) ); + nameLabel->setText( i18n( "Database name:" ) ); + portLabel->setText( i18n( "Port:" ) ); + portEdit->setSpecialValueText( i18n("Default") ); + + //set defaults + hostEdit->setText( "localhost" ); + nameEdit->setText( "Krecipes" ); + + // get username + uid_t userID; + struct passwd *user; + userID = getuid(); + user = getpwuid ( userID ); + TQString username(user->pw_name); + + userEdit->setText( username ); +} + +void DBImportDialog::switchDBPage( int id ) +{ + switch ( id ) { + case 0: //SQLite + paramStack->raiseWidget( sqlitePage ); + break; + case 1: //MySQL + case 2: //PostgreSQL + paramStack->raiseWidget( serverPage ); + break; + } +} + +TQString DBImportDialog::dbType() const +{ + int id = dbButtonGroup->id( dbButtonGroup->selected() ); + switch ( id ) { + case 0: + return "SQLite"; + case 1: + return "MySQL"; + case 2: + return "PostgreSQL"; + default: + return TQString::null; + } +} + +void DBImportDialog::serverParams( TQString &host, TQString &user, TQString &pass, int &port, TQString &table ) const +{ + host = hostEdit->text(); + user = userEdit->text(); + pass = passwordEdit->text(); + table = nameEdit->text(); + port = portEdit->value(); +} + +TQString DBImportDialog::dbFile() const +{ + return sqliteDBRequester->url(); +} + +#include "dbimportdialog.moc" diff --git a/src/dialogs/dbimportdialog.h b/src/dialogs/dbimportdialog.h new file mode 100644 index 0000000..eef49c5 --- /dev/null +++ b/src/dialogs/dbimportdialog.h @@ -0,0 +1,76 @@ +/*************************************************************************** +* Copyright (C) 2004 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ +#ifndef DBIMPORTDIALOG_H +#define DBIMPORTDIALOG_H + +#include <kdialogbase.h> + +class TQVBoxLayout; +class TQHBoxLayout; +class TQGridLayout; +class TQSpacerItem; +class TQButtonGroup; +class TQRadioButton; +class TQWidgetStack; +class TQWidget; +class TQLineEdit; +class TQLabel; +class TQPushButton; + +class KURLRequester; +class KIntNumInput; + +/** +@author Jason Kivlighn +*/ + +class DBImportDialog: public KDialogBase +{ + TQ_OBJECT +public: + DBImportDialog( TQWidget *parent = 0, const char *name = 0 ); + + TQString dbType() const; + void serverParams( TQString &host, TQString &user, TQString &pass, int &port, TQString &table ) const; + TQString dbFile() const; + +private: + // Widgets + TQButtonGroup* dbButtonGroup; + TQRadioButton* liteRadioButton; + TQRadioButton* mysqlRadioButton; + TQRadioButton* psqlRadioButton; + TQWidgetStack* paramStack; + TQWidget* serverPage; + TQLineEdit* nameEdit; + TQLabel* passwordLabel; + TQLineEdit* hostEdit; + TQLineEdit* passwordEdit; + TQLineEdit* userEdit; + TQLabel* userLabel; + TQLabel* hostLabel; + TQLabel* nameLabel; + TQLabel* portLabel; + KIntNumInput *portEdit; + TQWidget* sqlitePage; + KURLRequester* sqliteDBRequester; + + TQVBoxLayout* dbButtonGroupLayout; + TQVBoxLayout* serverPageLayout; + TQGridLayout* layout5; + TQVBoxLayout* serverPageLayout_2; + +protected slots: + void languageChange(); + void switchDBPage( int id ); + +}; + +#endif diff --git a/src/dialogs/dependanciesdialog.cpp b/src/dialogs/dependanciesdialog.cpp new file mode 100644 index 0000000..ac7d095 --- /dev/null +++ b/src/dialogs/dependanciesdialog.cpp @@ -0,0 +1,100 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* * +* Copyright (C) 2003-2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "dependanciesdialog.h" +#include "datablocks/elementlist.h" + +#include <tqvbox.h> + +#include <tdelocale.h> +#include <tdeglobal.h> +#include <tdeconfig.h> +#include <tdemessagebox.h> + +DependanciesDialog::DependanciesDialog( TQWidget *parent, const TQValueList<ListInfo> &lists, bool deps_are_deleted ) : KDialogBase( parent, "DependanciesDialog", true, TQString::null, + KDialogBase::Ok | KDialogBase::Cancel, KDialogBase::Cancel ), + m_depsAreDeleted(deps_are_deleted) +{ + init( lists ); +} + +DependanciesDialog::DependanciesDialog( TQWidget *parent, const ListInfo &list, bool deps_are_deleted ) : KDialogBase( parent, "DependanciesDialog", true, TQString::null, + KDialogBase::Ok | KDialogBase::Cancel, KDialogBase::Cancel ), + m_depsAreDeleted(deps_are_deleted) +{ + TQValueList<ListInfo> lists; + lists << list; + init( lists ); +} + +DependanciesDialog::~DependanciesDialog() +{} + +void DependanciesDialog::init( const TQValueList<ListInfo> &lists ) +{ + TQVBox *page = makeVBoxMainWidget(); + + // Design the dialog + + instructionsLabel = new TQLabel( page ); + instructionsLabel->setMinimumSize( TQSize( 100, 30 ) ); + instructionsLabel->setMaximumSize( TQSize( 10000, 10000 ) ); + instructionsLabel->setAlignment( int( TQLabel::WordBreak | TQLabel::AlignVCenter ) ); + + if ( m_depsAreDeleted ) { + instructionsLabel->setText( i18n( "<b>WARNING:</b> The following will have to be removed also, since currently they use the element you have chosen to be removed." ) ); + } + else { + instructionsLabel->setText( i18n( "<b>WARNING:</b> The following currently use the element you have chosen to be removed." ) ); + } + + for ( TQValueList<ListInfo>::const_iterator list_it = lists.begin(); list_it != lists.end(); ++list_it ) { + if ( !((*list_it).list).isEmpty() ) { + TQGroupBox *groupBox = new TQGroupBox( 1, TQt::Vertical, (*list_it).name, page ); + TDEListBox *listBox = new TDEListBox( groupBox ); + loadList( listBox, (*list_it).list ); + } + } + + setSizeGripEnabled( true ); +} + +void DependanciesDialog::loadList( TDEListBox* listBox, const ElementList &list ) +{ + TDEConfig * config = TDEGlobal::config(); + config->setGroup( "Advanced" ); + bool show_id = config->readBoolEntry( "ShowID", false ); + + for ( ElementList::const_iterator el_it = list.begin(); el_it != list.end(); ++el_it ) { + TQString name = ( *el_it ).name; + if ( show_id ) + name += " (" + TQString::number(( *el_it ).id) + ")"; + listBox->insertItem( name ); + } +} + +void DependanciesDialog::accept() +{ + if ( !m_msg.isEmpty() ) { + switch ( KMessageBox::warningYesNo(this, + TQString("<b>%1</b><br><br>%2").arg(m_msg).arg(i18n("Are you sure you wish to proceed?")), + TQString::null,KStdGuiItem::yes(),KStdGuiItem::no(),"doubleCheckDelete") ) + { + case KMessageBox::Yes: TQDialog::accept(); break; + case KMessageBox::No: TQDialog::reject(); break; + } + } + else + TQDialog::accept(); +} diff --git a/src/dialogs/dependanciesdialog.h b/src/dialogs/dependanciesdialog.h new file mode 100644 index 0000000..076bd04 --- /dev/null +++ b/src/dialogs/dependanciesdialog.h @@ -0,0 +1,60 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* * +* Copyright (C) 2003-2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef DEPENDANCIESDIALOG_H +#define DEPENDANCIESDIALOG_H + +#include <tqgroupbox.h> +#include <tqlabel.h> + +#include <tdelistview.h> +#include <kdialogbase.h> + +#include "datablocks/elementlist.h" + +struct ListInfo { + ElementList list; + TQString name; +}; + +/** +@author Unai Garro +*/ +class DependanciesDialog: public KDialogBase +{ +public: + //Methods + DependanciesDialog( TQWidget *parent, const TQValueList<ListInfo> &lists, bool deps_are_deleted = true ); + DependanciesDialog( TQWidget *parent, const ListInfo &list, bool deps_are_deleted = true ); + + ~DependanciesDialog(); + + void setCustomWarning( const TQString &msg ){ m_msg = msg; } + +public slots: + void accept(); + +private: + //Widgets + TQLabel *instructionsLabel; + + bool m_depsAreDeleted; + TQString m_msg; + + // Methods + void init( const TQValueList<ListInfo> &lists ); + void loadList( TDEListBox*, const ElementList &list ); +}; + +#endif diff --git a/src/dialogs/dietviewdialog.cpp b/src/dialogs/dietviewdialog.cpp new file mode 100644 index 0000000..5b51c27 --- /dev/null +++ b/src/dialogs/dietviewdialog.cpp @@ -0,0 +1,137 @@ +/*************************************************************************** +* Copyright (C) 2003-2005 * +* Unai Garro ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "dietviewdialog.h" + +#include <kiconloader.h> +#include <tdelocale.h> +#include <kstandarddirs.h> + +DietViewDialog::DietViewDialog( TQWidget *parent, const RecipeList &recipeList, int dayNumber, int mealNumber, const TQValueList <int> &dishNumbers ) + : KDialogBase( parent, "dietViewDialog", true, TQString::null, + KDialogBase::User2 | KDialogBase::Close | KDialogBase::User1, KDialogBase::User2, + false, KStdGuiItem::print() ) +{ + setButtonText( KDialogBase::User2, i18n( "Create &Shopping List" ) ); + + // Design the dialog + TQVBox *page = makeVBoxMainWidget(); + + // The html part + dietView = new TDEHTMLPart( page ); + + setInitialSize( TQSize(350, 450) ); + + setSizeGripEnabled( true ); + + connect ( this, TQ_SIGNAL( user2Clicked() ), this, TQ_SLOT( slotOk() ) ); + connect ( this, TQ_SIGNAL( closeClicked() ), this, TQ_SLOT( close() ) ); + connect ( this, TQ_SIGNAL( user1Clicked() ), this, TQ_SLOT( print() ) ); + + // Show the diet + showDiet( recipeList, dayNumber, mealNumber, dishNumbers ); +} + +DietViewDialog::~DietViewDialog() +{} + +void DietViewDialog::showDiet( const RecipeList &recipeList, int dayNumber, int mealNumber, const TQValueList <int> &dishNumbers ) +{ + + + // Header + TQString htmlCode = TQString( "<html><head><title>%1</title>" ).arg( i18n( "Diet" ) ); + + // CSS + htmlCode += "<STYLE type=\"text/css\">\n"; + htmlCode += "#calendar{border: thin solid black}"; + htmlCode += ".dayheader{ background-color: #D6D6D6; color: black; border:none;}"; + htmlCode += ".day{ background-color: #E5E5E5; color: black; border:medium solid #D6D6D6;}"; + htmlCode += ".meal{ background-color: #CDD4FF; color: black; border:thin solid #B4BEFF; text-align:center;}"; + htmlCode += ".dish{font-size: smaller; overflow: hidden; height:2.5em;}"; + htmlCode += "</STYLE>"; + + + htmlCode += "</head><body>"; // /Header + + // Calendar border + htmlCode += TQString( "<div id=\"calendar\">" ); + + // Title + htmlCode += TQString( "<center><div STYLE=\"width: 100%\">" ); + htmlCode += TQString( "<h1>%1</h1></div></center>" ).arg( i18n( "Diet" ) ); + + // Diet table + htmlCode += TQString( "<center><div STYLE=\"width: 98%\">" ); + htmlCode += TQString( "<table><tbody>" ); + + + TQValueList <int>::ConstIterator it; + it = dishNumbers.begin(); + RecipeList::ConstIterator rit; + rit = recipeList.begin(); + + for ( int row = 0, day = 0; row <= ( ( dayNumber - 1 ) / 7 ); row++ ) // New row (week) + { + htmlCode += TQString( "<tr>" ); + + for ( int col = 0; ( col < 7 ) && ( day < dayNumber ); col++, day++ ) // New column (day) + { + htmlCode += TQString( "<td><div class=\"day\">" ); + htmlCode += TQString( "<div class=\"dayheader\"><center>" ); + htmlCode += TQString( i18n( "Day %1" ) ).arg( day + 1 ); + htmlCode += TQString( "</center></div>" ); + for ( int meal = 0;meal < mealNumber;meal++ ) // Meals in each cell + { + int dishNumber = *it; + htmlCode += TQString( "<div class=\"meal\">" ); + for ( int dish = 0; dish < dishNumber;dish++ ) // Dishes in each Meal + { + htmlCode += TQString( "<div class=\"dish\">" ); + htmlCode += ( *rit ).title; + htmlCode += "<br>"; + htmlCode += TQString( "</div>" ); + rit++; + } + it++; + htmlCode += TQString( "</div>" ); + } + it = dishNumbers.begin(); // meals have same dish number everyday + htmlCode += TQString( "</div></td>" ); + } + + htmlCode += TQString( "</tr>" ); + } + + htmlCode += TQString( "</tbody></table>" ); + htmlCode += TQString( "</div></center>" ); + htmlCode += TQString( "</div></body></html>" ); + + resize( TQSize( 600, 400 ) ); + + // Display it + dietView->begin( KURL( locateLocal( "tmp", "/" ) ) ); // Initialize to tmp dir, where photos and logos can be stored + dietView->write( htmlCode ); + dietView->end(); +} + +void DietViewDialog::print( void ) +{ + dietView->view()->print(); +} + +void DietViewDialog::slotOk( void ) +{ + emit signalOk(); + close(); +} + +#include "dietviewdialog.moc" diff --git a/src/dialogs/dietviewdialog.h b/src/dialogs/dietviewdialog.h new file mode 100644 index 0000000..aff4992 --- /dev/null +++ b/src/dialogs/dietviewdialog.h @@ -0,0 +1,45 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef DIETVIEWDIALOG_H +#define DIETVIEWDIALOG_H + +#include <tqpushbutton.h> +#include <tqlayout.h> +#include <tqvbox.h> +#include <tqwidget.h> + +#include <tdehtml_part.h> +#include <tdehtmlview.h> +#include <kdialogbase.h> + +#include "datablocks/recipelist.h" + +class DietViewDialog: public KDialogBase +{ + TQ_OBJECT + +public: + DietViewDialog( TQWidget *parent, const RecipeList &recipeList, int dayNumber, int mealNumber, const TQValueList <int> &dishNumbers ); + ~DietViewDialog(); +private: + // Widgets + TDEHTMLPart *dietView; + + // Private methods + void showDiet( const RecipeList &recipeList, int dayNumber, int mealNumber, const TQValueList <int> &dishNumbers ); +private slots: + void print( void ); + void slotOk( void ); +signals: + void signalOk( void ); +}; + +#endif diff --git a/src/dialogs/dietwizarddialog.cpp b/src/dialogs/dietwizarddialog.cpp new file mode 100644 index 0000000..30414cf --- /dev/null +++ b/src/dialogs/dietwizarddialog.cpp @@ -0,0 +1,748 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "dietwizarddialog.h" +#include "backends/recipedb.h" +#include "dietviewdialog.h" + +#include <tqbitmap.h> +#include <tqheader.h> +#include <tqlayout.h> +#include <tqpainter.h> +#include <tqwmatrix.h> + +#include <tdeapplication.h> +#include <kcursor.h> +#include <tdeglobalsettings.h> +#include <kiconloader.h> +#include <tdelocale.h> +#include <tdemessagebox.h> +#include <knuminput.h> +#include <kdebug.h> + +#include "propertycalculator.h" +#include "widgets/propertylistview.h" +#include "widgets/categorylistview.h" + +#include "profiling.h" + +DietWizardDialog::DietWizardDialog( TQWidget *parent, RecipeDB *db ) : TQVBox( parent ) +{ + // Initialize internal variables + database = db; + mealNumber = 1; + dayNumber = 1; + dietRList = new RecipeList(); + + //Design the dialog + setSpacing( 5 ); + // Options Box + optionsBox = new TQHBox( this ); + + daysSliderBox = new TQVGroupBox( i18n( "Number of Days" ), optionsBox ); + dayNumberLabel = new TQLabel( daysSliderBox ); + dayNumberLabel->setText( "- 1 -" ); + dayNumberLabel->setAlignment( TQt::AlignHCenter ); + dayNumberSelector = new TQSlider( daysSliderBox ); + + dayNumberSelector->setOrientation( TQt::Horizontal ); + dayNumberSelector->setRange( 1, 10 ); + dayNumberSelector->setSteps( 1, 1 ); + dayNumberSelector->setTickmarks( TQSlider::Below ); + dayNumberSelector->setFixedWidth( 100 ); + + mealsSliderBox = new TQVGroupBox( i18n( "Meals per Day" ), optionsBox ); + mealNumberLabel = new TQLabel( mealsSliderBox ); + mealNumberLabel->setText( "- 1 -" ); + mealNumberLabel->setAlignment( TQt::AlignHCenter ); + mealNumberSelector = new TQSlider( mealsSliderBox ); + + mealNumberSelector->setOrientation( TQt::Horizontal ); + mealNumberSelector->setRange( 1, 10 ); + mealNumberSelector->setSteps( 1, 1 ); + mealNumberSelector->setTickmarks( TQSlider::Below ); + mealNumberSelector->setFixedWidth( 100 ); + + // Tabs + mealTabs = new TQTabWidget( this ); + mealTabs->setMargin( 5 ); + + // Button bar + TDEIconLoader il; + + TQHBox *bottom_layout = new TQHBox( this ); + //bottom_layout->layout()->addItem( new TQSpacerItem( 10,10, TQSizePolicy::MinimumExpanding, TQSizePolicy::Fixed ) ); + + okButton = new TQPushButton( bottom_layout ); + okButton->setIconSet( il.loadIconSet( "button_ok", TDEIcon::Small ) ); + okButton->setText( i18n( "Create the diet" ) ); + + TQPushButton *clearButton = new TQPushButton( bottom_layout ); + clearButton->setIconSet( il.loadIconSet( "edit-clear", TDEIcon::Small ) ); + clearButton->setText( i18n( "Clear" ) ); + + // Create Tabs + //don't use newTab, it'll load data and we don't want it to do that at startup + mealTab = new MealInput( mealTabs, database ); + mealTabs->addTab( mealTab,i18n( "Meal 1" ) ); + mealTabs->setCurrentPage( mealTabs->indexOf( mealTab ) ); + + // Signals & Slots + connect( mealNumberSelector, TQ_SIGNAL( valueChanged( int ) ), this, TQ_SLOT( changeMealNumber( int ) ) ); + connect( dayNumberSelector, TQ_SIGNAL( valueChanged( int ) ), this, TQ_SLOT( changeDayNumber( int ) ) ); + connect( okButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( createDiet() ) ); + connect( clearButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( clear() ) ); +} + + +DietWizardDialog::~DietWizardDialog() +{ + delete dietRList; +} + +void DietWizardDialog::clear() +{ + mealNumberSelector->setValue( 1 ); + dayNumberSelector->setValue( 1 ); + + MealInput* mealTab = ( MealInput* ) ( mealTabs->page( 0 ) ); // Get the meal + mealTab->setDishNo( 3 ); + mealTab->showDish( 0 ); + + for ( uint i = 0; i < mealTab->dishInputList.count(); ++i ) { + DishInput* dishInput = mealTab->dishInputList[ i ]; // Get the dish input + dishInput->clear(); + } +} + +void DietWizardDialog::reload( ReloadFlags flag ) +{ + for ( int i = 0; i < mealTabs->count(); ++i ) { + MealInput *mealTab = (MealInput*)mealTabs->page(i); + mealTab->reload(flag); + } +} + +void DietWizardDialog::newTab( const TQString &name ) +{ + mealTab = new MealInput( mealTabs, database ); + mealTab->reload(); + mealTabs->addTab( mealTab, name ); + mealTabs->setCurrentPage( mealTabs->indexOf( mealTab ) ); +} + +void DietWizardDialog::changeMealNumber( int mn ) +{ + mealNumberLabel->setText( TQString( i18n( "- %1 -" ) ).arg( mn ) ); + if ( mn > mealNumber ) { + + while ( mealNumber != mn ) { + mealNumber++; + newTab( i18n( "Meal %1" ).arg( mealNumber ) ); + + } + } + else if ( mn < mealNumber ) { + + while ( mealNumber != mn ) { + mealNumber--; + delete mealTabs->page( mealTabs->count() - 1 ); + } + } +} + +void DietWizardDialog::changeDayNumber( int dn ) +{ + + if ( dn < 7 ) { + dayNumber = dn; + dayNumberLabel->setText( TQString( i18n( "- %1 -" ) ).arg( dn ) ); + } + else if ( dn == 7 ) { + dayNumber = 7; + dayNumberLabel->setText( TQString( i18n( "- 1 week -" ) ) ); + } + else { + dayNumber = ( dn - 6 ) * 7; + dayNumberLabel->setText( TQString( i18n( "- %1 weeks -" ) ).arg( dn - 6 ) ); + } +} + +void DietWizardDialog::createDiet( void ) +{ + TDEApplication::setOverrideCursor( KCursor::waitCursor() ); + + START_TIMER("Creating the diet"); + + RecipeList rlist; + dietRList->clear(); + + // Get the whole list of recipes, detailed + int flags = RecipeDB::Title | getNecessaryFlags(); + database->loadRecipes( &rlist, flags ); + + // temporal iterator list so elements can be removed without reloading them again from the DB + // this list prevents the same meal from showing up in the same day twice + TQValueList <RecipeList::Iterator> tempRList; + + bool alert = false; + + for ( int day = 0;day < dayNumber;day++ ) // Create the diet for the number of days defined by the user + { + populateIteratorList( rlist, &tempRList ); // temporal iterator list so elements can be removed without reloading them again from the DB + for ( int meal = 0;meal < mealNumber;meal++ ) + { + int dishNo = ( ( MealInput* ) ( mealTabs->page( meal ) ) ) ->dishNo(); + + for ( int dish = 0;dish < dishNo;dish++ ) { + bool found = false; + TQValueList <RecipeList::Iterator> tempDishRList = tempRList; + while ( ( !found ) && !tempDishRList.empty() ) { + int random_index = ( int ) ( ( float ) ( kapp->random() ) / ( float ) RAND_MAX * tempDishRList.count() ); + TQValueList<RecipeList::Iterator>::Iterator iit = tempDishRList.at( random_index ); // note that at() retrieves an iterator to the iterator list, so we need to use * in order to get the RecipeList::Iterator + + RecipeList::Iterator rit = *iit; + if ( found = ( ( ( !categoryFiltering( meal, dish ) ) || checkCategories( *rit, meal, dish ) ) && checkConstraints( *rit, meal, dish ) ) ) // Check that the recipe is inside the constraint limits and in the categories specified + { + dietRList->append( *rit ); // Add recipe to the diet list + tempRList.remove( tempRList.find(*iit) ); //can't just remove()... the iterator isn't from this list (its an iterator from tempDishRList) + } + else { + tempDishRList.remove( iit ); // Remove this analized recipe from teh list + } + } + if ( !found ) + alert = true; + } + } + } + + if ( alert ) { + TDEApplication::restoreOverrideCursor(); + KMessageBox::sorry( this, i18n( "I could not create a full diet list given the constraints. Either the recipe list is too short or the constraints are too demanding. " ) ); + } + + else // show the resulting diet + { + + // make a list of dishnumbers + TQValueList<int> dishNumbers; + + for ( int meal = 0;meal < mealNumber;meal++ ) { + int dishNo = ( ( MealInput* ) ( mealTabs->page( meal ) ) ) ->dishNo(); + dishNumbers << dishNo; + } + + TDEApplication::restoreOverrideCursor(); + + // display the list + DietViewDialog dietDisplay( this, *dietRList, dayNumber, mealNumber, dishNumbers ); + connect( &dietDisplay, TQ_SIGNAL( signalOk() ), this, TQ_SLOT( createShoppingList() ) ); + dietDisplay.exec(); + } + + END_TIMER(); +} + + +void DietWizardDialog::populateIteratorList( RecipeList &rl, TQValueList <RecipeList::Iterator> *il ) +{ + il->clear(); + RecipeList::Iterator it; + for ( it = rl.begin();it != rl.end(); ++it ) + il->append( it ); + +} + +int DietWizardDialog::getNecessaryFlags() const +{ + bool need_ingredients = false; + bool need_categories = false; + for ( int meal = 0;meal < mealNumber;meal++ ) { + int dishNo = ( ( MealInput* ) ( mealTabs->page( meal ) ) ) ->dishNo(); + for ( int dish = 0;dish < dishNo;dish++ ) { + if ( !need_categories ) { + if ( categoryFiltering( meal, dish ) ) { + need_categories = true; + } + } + + if ( !need_ingredients ) { + ConstraintList constraints; + loadConstraints( meal, dish, &constraints ); + for ( ConstraintList::const_iterator ct_it = constraints.begin(); ct_it != constraints.end(); ++ct_it ) { + if ( (*ct_it).enabled ) { + need_ingredients = true; + break; + } + } + } + + if ( need_ingredients && need_categories ) + break; + } + + if ( need_ingredients && need_categories ) + break; + } + + kdDebug()<<"Do we need to load ingredients: "<<need_ingredients<<endl; + kdDebug()<<"Do we need to load categories: "<<need_categories<<endl; + + int flags = 0; + if ( need_ingredients ) flags |= RecipeDB::Ingredients; + if ( need_categories ) flags |= RecipeDB::Categories; + + return flags; +} + + +MealInput::MealInput( TQWidget *parent, RecipeDB *db ) : TQWidget( parent ), + database( db ) +{ + // Design the dialog + TQVBoxLayout *layout = new TQVBoxLayout( this ); + layout->setSpacing( 10 ); + + // Options box + + mealOptions = new TQHBox( this ); + mealOptions->setSpacing( 10 ); + layout->addWidget( mealOptions ); + + // Number of dishes input + dishNumberBox = new TQHBox( mealOptions ); + dishNumberBox->setSpacing( 10 ); + dishNumberLabel = new TQLabel( i18n( "No. of dishes: " ), dishNumberBox ); + dishNumberInput = new TQSpinBox( dishNumberBox ); + dishNumberInput->setMinValue( 1 ); + dishNumberInput->setMaxValue( 10 ); + dishNumberBox->setSizePolicy( TQSizePolicy( TQSizePolicy::Minimum, TQSizePolicy::Minimum ) ); + + // Toolbar + + toolBar = new TQHGroupBox( mealOptions ); + toolBar->setFrameStyle ( TQFrame::Panel | TQFrame::Sunken ); + toolBar->setSizePolicy( TQSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::Minimum ) ); + + // Next dish/ Previous dish buttons + TDEIconLoader il; + buttonPrev = new TQToolButton( toolBar ); + buttonPrev->setUsesTextLabel( true ); + buttonPrev->setTextLabel( i18n( "Previous Dish" ) ); + buttonPrev->setIconSet( il.loadIconSet( "back", TDEIcon::Small ) ); + buttonPrev->setTextPosition( TQToolButton::Under ); + buttonNext = new TQToolButton( toolBar ); + buttonNext->setUsesTextLabel( true ); + buttonNext->setTextLabel( i18n( "Next Dish" ) ); + buttonNext->setIconSet( il.loadIconSet( "forward", TDEIcon::Small ) ); + buttonNext->setTextPosition( TQToolButton::Under ); + + + // Dish widgets + dishStack = new TQWidgetStack( this ); + layout->addWidget( dishStack ); + dishStack->setSizePolicy( TQSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::MinimumExpanding ) ); + // Add default dishes + DishInput *newDish = new DishInput( this, database, i18n( "1st Course" ) ); + dishStack->addWidget( newDish ); + dishInputList.append( newDish ); + newDish = new DishInput( this, database, i18n( "2nd Course" ) ); + dishStack->addWidget( newDish ); + dishInputList.append( newDish ); + newDish = new DishInput( this, database, i18n( "Dessert" ) ); + dishStack->addWidget( newDish ); + dishInputList.append( newDish ); + dishNumber = 3; + dishNumberInput->setValue( dishNumber ); + + // Signals & Slots + connect( dishNumberInput, TQ_SIGNAL( valueChanged( int ) ), this, TQ_SLOT( changeDishNumber( int ) ) ); + connect( buttonPrev, TQ_SIGNAL( clicked() ), this, TQ_SLOT( prevDish() ) ); + connect( buttonNext, TQ_SIGNAL( clicked() ), this, TQ_SLOT( nextDish() ) ); + +} + +MealInput::~MealInput() +{} + +void MealInput::reload( ReloadFlags flag ) +{ + TQValueList<DishInput*>::iterator it; + for ( it = dishInputList.begin(); it != dishInputList.end(); ++it ) { + DishInput *di = ( *it ); + di->reload(flag); + } +} + +void MealInput::setDishNo( int dn ) +{ + dishNumberInput->setValue( dn ); +} + +void MealInput::changeDishNumber( int dn ) +{ + if ( dn > dishNumber ) { + while ( dishNumber != dn ) { + DishInput * newDish = new DishInput( this, database, TQString( i18n( "Dish %1" ) ).arg( dishNumber + 1 ) ); + newDish->reload(); + dishStack->addWidget( newDish ); + dishInputList.append( newDish ); + dishStack->raiseWidget( newDish ); + dishNumber++; + } + } + else if ( dn < dishNumber ) { + TQValueList <DishInput*>::Iterator it; + while ( dishNumber != dn ) { + it = dishInputList.fromLast(); + DishInput *lastDish = ( *it ); + dishInputList.remove( it ); + + if ( ( *it ) == ( DishInput* ) dishStack->visibleWidget() ) { + it--; + dishStack->raiseWidget( *it ); + } + delete lastDish; + dishNumber--; + } + } +} + + +void MealInput::nextDish( void ) +{ + // First get the place of the current dish input in the list + TQValueList <DishInput*>::iterator it = dishInputList.find( ( DishInput* ) ( dishStack->visibleWidget() ) ); + + //Show the next dish if it exists + it++; + if ( it != dishInputList.end() ) { + dishStack->raiseWidget( *it ); + } + +} + +void MealInput::prevDish( void ) +{ + // First get the place of the current dish input in the list + TQValueList <DishInput*>::iterator it = dishInputList.find( ( DishInput* ) ( dishStack->visibleWidget() ) ); + + //Show the previous dish if it exists + it--; + if ( it != dishInputList.end() ) { + dishStack->raiseWidget( *it ); + } +} + +void MealInput::showDish( int dn ) +{ + TQValueList <DishInput*>::iterator it = dishInputList.at( dn ); + if ( it != dishInputList.end() ) + dishStack->raiseWidget( *it ); +} + +DishInput::DishInput( TQWidget* parent, RecipeDB *db, const TQString &title ) : TQWidget( parent ), + database(db) +{ + + // Initialize internal variables + categoryFiltering = false; + + // Design the widget + + TQVBoxLayout *layout = new TQVBoxLayout( this ); + layout->setSpacing( 10 ); + + //Horizontal Box to hold the TDEListView's + listBox = new TQHGroupBox( i18n( "Dish Characteristics" ), this ); + layout->addWidget( listBox ); + + // Dish id + dishTitle = new DishTitle( listBox, title ); + + //Categories list + categoriesBox = new TQVBox( listBox ); + categoriesEnabledBox = new TQCheckBox( categoriesBox ); + categoriesEnabledBox->setText( i18n( "Enable Category Filtering" ) ); + + categoriesView = new CategoryCheckListView( categoriesBox, database, false ); + categoriesView->setSizePolicy( TQSizePolicy::Ignored, TQSizePolicy::Ignored ); + categoriesView->setEnabled( false ); // Disable it by default + + //Constraints list + constraintsView = new PropertyConstraintListView( listBox, database ); + constraintsView->setSizePolicy( TQSizePolicy::Ignored, TQSizePolicy::Ignored ); + constraintsView->reload(); + + // KDoubleInput based edit boxes + constraintsEditBox1 = new KDoubleNumInput( constraintsView->viewport() ); + constraintsView->addChild( constraintsEditBox1 ); + constraintsEditBox1->hide(); + constraintsEditBox2 = new KDoubleNumInput( constraintsView->viewport() ); + constraintsView->addChild( constraintsEditBox2 ); + constraintsEditBox2->hide(); + + + // Connect Signals & Slots + connect( constraintsView, TQ_SIGNAL( executed( TQListViewItem* ) ), this, TQ_SLOT( insertConstraintsEditBoxes( TQListViewItem* ) ) ); + connect( constraintsView, TQ_SIGNAL( selectionChanged() ), this, TQ_SLOT( hideConstraintInputs() ) ); + connect( constraintsEditBox1, TQ_SIGNAL( valueChanged( double ) ), this, TQ_SLOT( setMinValue( double ) ) ); + connect( constraintsEditBox2, TQ_SIGNAL( valueChanged( double ) ), this, TQ_SLOT( setMaxValue( double ) ) ); + connect( categoriesEnabledBox, TQ_SIGNAL( toggled( bool ) ), this, TQ_SLOT( enableCategories( bool ) ) ); +} + +DishInput::~DishInput() +{} + +void DishInput::clear() +{ + TQListViewItemIterator it( categoriesView ); + while ( it.current() ) { + TQCheckListItem * item = ( TQCheckListItem* ) it.current(); + item->setOn( false ); + ++it; + } + + constraintsView->reload(); + categoriesEnabledBox->setChecked( false ); +} + +void DishInput::enableCategories( bool enable ) +{ + categoriesView->setEnabled( enable ); + categoryFiltering = enable; +} + +bool DishInput::isCategoryFilteringEnabled( void ) const +{ + return categoryFiltering; +} + +void DishInput::reload( ReloadFlags flag ) +{ + constraintsView->reload(); + categoriesView->reload(flag); +} + +void DishInput::insertConstraintsEditBoxes( TQListViewItem* it ) +{ + TQRect r; + + // Constraints Box1 + r = constraintsView->header() ->sectionRect( 2 ); //start at the section 2 header + r.moveBy( 0, constraintsView->itemRect( it ).y() ); //Move down to the item, note that its height is same as header's right now. + + r.setHeight( it->height() ); // Set the item's height + r.setWidth( constraintsView->header() ->sectionRect( 2 ).width() ); // and width + constraintsEditBox1->setGeometry( r ); + + + //Constraints Box2 + r = constraintsView->header() ->sectionRect( 3 ); //start at the section 3 header + r.moveBy( 0, constraintsView->itemRect( it ).y() ); //Move down to the item + + r.setHeight( it->height() ); // Set the item's height + r.setWidth( constraintsView->header() ->sectionRect( 3 ).width() ); // and width + constraintsEditBox2->setGeometry( r ); + + // Set the values from the item + constraintsEditBox1->setValue( ( ( ConstraintsListItem* ) it ) ->minVal() ); + constraintsEditBox2->setValue( ( ( ConstraintsListItem* ) it ) ->maxVal() ); + + // Show Boxes + constraintsEditBox1->show(); + constraintsEditBox2->show(); +} + +void DishInput::hideConstraintInputs() +{ + constraintsEditBox1->hide(); + constraintsEditBox2->hide(); +} + +void DishInput::loadConstraints( ConstraintList *constraints ) const +{ + constraints->clear(); + Constraint constraint; + for ( ConstraintsListItem * it = ( ConstraintsListItem* ) ( constraintsView->firstChild() );it;it = ( ConstraintsListItem* ) ( it->nextSibling() ) ) { + constraint.id = it->propertyId(); + constraint.min = it->minVal(); + constraint.max = it->maxVal(); + constraint.enabled = it->isOn(); + constraints->append( constraint ); + } +} + +void DishInput::loadEnabledCategories( ElementList* categories ) +{ + categories->clear(); + + // Only load those that are checked, unless filtering is disabled + if ( !categoriesView->isEnabled() ) { + database->loadCategories(categories); + } + else { + *categories = categoriesView->selections(); + } +} + +void DishInput::setMinValue( double minValue ) +{ + ConstraintsListItem *it = ( ConstraintsListItem* ) ( constraintsView->selectedItem() ); // Find selected property + + if ( it ) + it->setMin( minValue ); +} + +void DishInput::setMaxValue( double maxValue ) +{ + ConstraintsListItem *it = ( ConstraintsListItem* ) ( constraintsView->selectedItem() ); // Find selected property + + if ( it ) + it->setMax( maxValue ); +} + +DishTitle::DishTitle( TQWidget *parent, const TQString &title ) : TQWidget( parent ) +{ + titleText = title; +} + + +DishTitle::~DishTitle() +{} + +void DishTitle::paintEvent( TQPaintEvent * ) +{ + // Now draw the text + TQPainter painter( this ); + + // First draw the decoration + painter.setPen( TDEGlobalSettings::activeTitleColor() ); + painter.setBrush( TQBrush( TDEGlobalSettings::activeTitleColor() ) ); + painter.drawRoundRect( 0, 20, 30, height() - 40, 50, ( int ) ( 50.0 / ( height() - 40 ) * 35.0 ) ); + + // Now draw the text + + TQFont titleFont = TDEGlobalSettings::windowTitleFont (); + titleFont.setPointSize( 15 ); + painter.setFont( titleFont ); + painter.rotate( -90 ); + painter.setPen( TQColor( 0x00, 0x00, 0x00 ) ); + painter.drawText( 0, 0, -height(), 30, AlignCenter, titleText ); + painter.setPen( TQColor( 0xFF, 0xFF, 0xFF ) ); + painter.drawText( -1, -1, -height() - 1, 29, AlignCenter, titleText ); + painter.end(); +} + +TQSize DishTitle::sizeHint () const +{ + return ( TQSize( 40, 200 ) ); +} + +TQSize DishTitle::minimumSizeHint() const +{ + return ( TQSize( 40, 200 ) ); +} + +bool DietWizardDialog::checkCategories( Recipe &rec, int meal, int dish ) +{ + + // Check if the recipe is among the categories chosen + ElementList categoryList; + loadEnabledCategories( meal, dish, &categoryList ); + + + for ( ElementList::const_iterator cat_it = rec.categoryList.begin(); cat_it != rec.categoryList.end(); ++cat_it ) { + if ( categoryList.containsId( ( *cat_it ).id ) ) + return true; + } + + return false; +} + +/* +** Calculate the recipe Properties and check if they're within the constraints +*/ + +bool DietWizardDialog::checkConstraints( Recipe &rec, int meal, int dish ) +{ + // Check if the properties are within the constraints + ConstraintList constraints; + loadConstraints( meal, dish, &constraints ); //load the constraints + + bool any_enabled = false; + for ( ConstraintList::const_iterator ct_it = constraints.begin(); ct_it != constraints.end(); ++ct_it ) { + if ( (*ct_it).enabled ) { + any_enabled = true; + break; + } + } + if ( !any_enabled ) + return true; + + // Calculate properties of the recipe + calculateProperties( rec, database ); + + bool withinLimits = checkLimits( rec.properties, constraints ); + + return ( withinLimits ); +} + +void DietWizardDialog::loadConstraints( int meal, int dish, ConstraintList *constraints ) const +{ + MealInput * mealTab = ( MealInput* ) ( mealTabs->page( meal ) ); // Get the meal + DishInput* dishInput = mealTab->dishInputList[ dish ]; // Get the dish input + dishInput->loadConstraints( constraints ); //Load the constraints form the TDEListView +} + +void DietWizardDialog::loadEnabledCategories( int meal, int dish, ElementList *categories ) +{ + MealInput * mealTab = ( MealInput* ) ( mealTabs->page( meal ) ); // Get the meal + DishInput* dishInput = mealTab->dishInputList[ dish ]; // Get the dish input + dishInput->loadEnabledCategories( categories ); //Load the categories that have been checked in the TDEListView +} + +bool DietWizardDialog::categoryFiltering( int meal, int dish ) const +{ + MealInput * mealTab = ( MealInput* ) ( mealTabs->page( meal ) ); // Get the meal + DishInput* dishInput = mealTab->dishInputList[ dish ]; // Get the dish input + return ( dishInput->isCategoryFilteringEnabled() ); //Load the categories that have been checked in the TDEListView +} + +bool DietWizardDialog::checkLimits( IngredientPropertyList &properties, ConstraintList &constraints ) +{ + for ( ConstraintList::const_iterator ct_it = constraints.begin(); ct_it != constraints.end(); ++ct_it ) { + if ( (*ct_it).enabled ) { + IngredientPropertyList::const_iterator ip_it = properties.find( (*ct_it).id ); + if ( ip_it != properties.end() ) { + if ( ( (*ip_it).amount > (*ct_it).max ) || ( (*ip_it).amount < (*ct_it).min ) ) + return false; + } + else + return false; + } + } + return true; +} + +void DietWizardDialog::createShoppingList( void ) +{ + emit dietReady(); +} + +RecipeList& DietWizardDialog::dietList( void ) +{ + return *dietRList; +} + +#include "dietwizarddialog.moc" diff --git a/src/dialogs/dietwizarddialog.h b/src/dialogs/dietwizarddialog.h new file mode 100644 index 0000000..c848664 --- /dev/null +++ b/src/dialogs/dietwizarddialog.h @@ -0,0 +1,222 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef DIETWIZARDDIALOG_H +#define DIETWIZARDDIALOG_H + +#include <stdlib.h> // For RAND_MAX + +#include <tqcheckbox.h> +#include <tqhbox.h> +#include <tqhgroupbox.h> +#include <tqlabel.h> +#include <tqptrlist.h> +#include <tqpushbutton.h> +#include <tqslider.h> +#include <tqspinbox.h> +#include <tqtabwidget.h> +#include <tqtoolbutton.h> +#include <tqvaluelist.h> +#include <tqvbox.h> +#include <tqvgroupbox.h> +#include <tqwidgetstack.h> + +#include <tdelistview.h> + +#include "datablocks/constraintlist.h" +#include "datablocks/recipelist.h" +#include "datablocks/elementlist.h" +#include "datablocks/ingredientpropertylist.h" +#include "datablocks/recipe.h" +#include "datablocks/unitratiolist.h" + +#include "widgets/dblistviewbase.h" + +class KDoubleNumInput; + +class DishInput; +class DishTitle; +class MealInput; +class RecipeDB; +class RecipeList; +class CategoryCheckListView; +class PropertyConstraintListView; + +/** +@author Unai Garro +*/ + + +class DietWizardDialog: public TQVBox +{ + + TQ_OBJECT + +public: + + DietWizardDialog( TQWidget *parent, RecipeDB* db ); + ~DietWizardDialog(); + +private: + //Private variables + RecipeDB *database; + + int dayNumber; + int mealNumber; + + RecipeList *dietRList; + + //Widgets + TQHBox *optionsBox; + TQVGroupBox *mealsSliderBox; + TQLabel *mealNumberLabel; + TQSlider *mealNumberSelector; + TQVGroupBox *daysSliderBox; + TQLabel *dayNumberLabel; + TQSlider *dayNumberSelector; + TQTabWidget *mealTabs; + MealInput *mealTab; // points to the current tab + TQPushButton *okButton; + + //Methods + bool checkCategories( Recipe &rec, int meal, int dish ); + bool checkConstraints( Recipe &rec, int meal, int dish ); + bool checkLimits( IngredientPropertyList &properties, ConstraintList &constraints ); + void loadConstraints( int meal, int dish, ConstraintList *constraints ) const; + void loadEnabledCategories( int meal, int dish, ElementList *categories ); + void newTab( const TQString &name ); + bool categoryFiltering( int meal, int dish ) const; + int getNecessaryFlags() const; + +public: + //Methods + void reload( ReloadFlags flags = Load ); + RecipeList& dietList( void ); + +private slots: + void changeDayNumber( int dn ); + void changeMealNumber( int mn ); + void createDiet( void ); + void clear(); + void createShoppingList( void ); + void populateIteratorList( RecipeList &rl, TQValueList <RecipeList::Iterator> *il ); +signals: + void dietReady( void ); +}; + +class MealInput: public TQWidget +{ + TQ_OBJECT + +public: + // Methods + + MealInput( TQWidget *parent, RecipeDB *database ); + ~MealInput(); + void reload( ReloadFlags flag = Load ); + int dishNo( void ) + { + return dishNumber; + }; + void setDishNo( int dn ); + void showDish( int dn ); + + // Public widgets and variables + TQValueList <DishInput*> dishInputList; // The list of dishes + +private: + // Widgets + // Private Variables + int dishNumber; + RecipeDB *database; + + // Settings section for the meal + TQHBox *mealOptions; + + // Dish number setting + TQHBox *dishNumberBox; + TQLabel *dishNumberLabel; + TQSpinBox *dishNumberInput; + + // Move <-> buttons + TQHGroupBox *toolBar; + TQToolButton *buttonNext; + TQToolButton *buttonPrev; + + // Settings for the dish + TQWidgetStack *dishStack; + +public slots: + void nextDish( void ); + void prevDish( void ); + +private slots: + void changeDishNumber( int dn ); + +}; + +class DishInput: public TQWidget +{ + TQ_OBJECT + +public: + DishInput( TQWidget *parent, RecipeDB *database, const TQString &title ); + ~DishInput(); + // Methods + bool isCategoryFilteringEnabled( void ) const; + void loadConstraints( ConstraintList *constraints ) const; + void loadEnabledCategories( ElementList* categories ); + void reload( ReloadFlags flag = Load ); + void setDishTitle( const TQString & text ); + void clear(); + + +private: + // Variables + bool categoryFiltering; + // Widgets + TQHGroupBox *listBox; + DishTitle *dishTitle; + TQVBox *categoriesBox; + TQCheckBox *categoriesEnabledBox; + CategoryCheckListView *categoriesView; + PropertyConstraintListView *constraintsView; + KDoubleNumInput *constraintsEditBox1; + KDoubleNumInput *constraintsEditBox2; + RecipeDB *database; + +private slots: + void enableCategories( bool enable ); + void insertConstraintsEditBoxes( TQListViewItem* it ); + void hideConstraintInputs(); + void setMinValue( double minValue ); + void setMaxValue( double maxValue ); +}; + +class DishTitle: public TQWidget +{ + + TQ_OBJECT + +public: + DishTitle( TQWidget *parent, const TQString &title ); + ~DishTitle(); + virtual TQSize sizeHint () const; + virtual TQSize minimumSizeHint() const; +protected: + //Variables + TQString titleText; + //Methods + virtual void paintEvent( TQPaintEvent *p ); +}; + +#endif diff --git a/src/dialogs/editratingdialog.cpp b/src/dialogs/editratingdialog.cpp new file mode 100644 index 0000000..bab6e1b --- /dev/null +++ b/src/dialogs/editratingdialog.cpp @@ -0,0 +1,243 @@ +/*************************************************************************** +* Copyright (C) 2005 by Jason Kivlighn * +* [email protected] * +* * +* 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. * +***************************************************************************/ + +#include "editratingdialog.h" + +#include <tdelocale.h> + +#include <tqvariant.h> +#include <tqpushbutton.h> +#include <tqlabel.h> +#include <tqcombobox.h> +#include <knuminput.h> +#include <tqheader.h> +#include <tdelistview.h> +#include <tqtextedit.h> +#include <tqlineedit.h> +#include <tqlayout.h> +#include <tqtooltip.h> +#include <tqwhatsthis.h> +#include <tqpainter.h> +#include <tqvbox.h> + +#include <tdepopupmenu.h> +#include <tdelocale.h> +#include <kiconloader.h> +#include <kdebug.h> + +#include "datablocks/rating.h" +#include "datablocks/elementlist.h" +#include "datablocks/mixednumber.h" + +#include "widgets/ratingwidget.h" + +class RatingCriteriaListView : public TDEListView +{ +public: + RatingCriteriaListView( TQWidget *parent = 0, const char *name = 0 ) : TDEListView(parent,name){} + + void rename( TQListViewItem *it, int c ) + { + if ( c == 1 ) + it->setPixmap(c,TQPixmap()); + + TDEListView::rename(it,c); + } +}; + + +EditRatingDialog::EditRatingDialog( const ElementList &criteriaList, const Rating &rating, TQWidget* parent, const char* name ) + : KDialogBase( parent, name, true, i18n( "Rating" ), + KDialogBase::Ok | KDialogBase::Cancel, KDialogBase::Ok ) +{ + init(criteriaList); + loadRating(rating); +} + +/* + * Constructs a EditRatingDialog as a child of 'parent', with the + * name 'name' and widget flags set to 'f'. + */ +EditRatingDialog::EditRatingDialog( const ElementList &criteriaList, TQWidget* parent, const char* name ) + : KDialogBase( parent, name, true, i18n( "Rating" ), + KDialogBase::Ok | KDialogBase::Cancel, KDialogBase::Ok ) +{ + init(criteriaList); +} + +void EditRatingDialog::init( const ElementList &criteriaList ) +{ + TQVBox *page = makeVBoxMainWidget(); + + layout2 = new TQHBox( page ); + + raterLabel = new TQLabel( layout2, "raterLabel" ); + raterEdit = new TQLineEdit( layout2, "raterEdit" ); + + layout8 = new TQHBox( page ); + + criteriaLabel = new TQLabel( layout8, "criteriaLabel" ); + + criteriaComboBox = new TQComboBox( FALSE, layout8, "criteriaComboBox" ); + criteriaComboBox->setSizePolicy( TQSizePolicy( TQSizePolicy::MinimumExpanding, (TQSizePolicy::SizeType)0, 0, 0, criteriaComboBox->sizePolicy().hasHeightForWidth() ) ); + criteriaComboBox->setEditable( TRUE ); + criteriaComboBox->lineEdit()->disconnect( criteriaComboBox ); //so hitting enter doesn't enter the item into the box + + starsLabel = new TQLabel( layout8, "starsLabel" ); + + starsWidget = new RatingWidget( 5, layout8, "starsWidget" ); + + addButton = new TQPushButton( layout8, "addButton" ); + + removeButton = new TQPushButton( layout8, "removeButton" ); + + criteriaListView = new RatingCriteriaListView( page, "criteriaListView" ); + criteriaListView->addColumn( i18n( "Criteria" ) ); + criteriaListView->addColumn( i18n( "Stars" ) ); + criteriaListView->setSizePolicy( TQSizePolicy( (TQSizePolicy::SizeType)7, (TQSizePolicy::SizeType)7, 0, 0, criteriaListView->sizePolicy().hasHeightForWidth() ) ); + criteriaListView->setSorting(-1); + criteriaListView->setItemsRenameable( true ); + criteriaListView->setRenameable( 0, true ); + criteriaListView->setRenameable( 1, true ); + + commentsLabel = new TQLabel( page, "commentsLabel" ); + + commentsEdit = new TQTextEdit( page, "commentsEdit" ); + + languageChange(); + resize( TQSize(358, 331).expandedTo(minimumSizeHint()) ); + clearWState( WState_Polished ); + + connect( criteriaListView, TQ_SIGNAL(itemRenamed(TQListViewItem*,const TQString &,int)), this, TQ_SLOT(itemRenamed(TQListViewItem*,const TQString &,int)) ); + connect( addButton, TQ_SIGNAL(clicked()), this, TQ_SLOT(slotAddRatingCriteria()) ); + connect( removeButton, TQ_SIGNAL(clicked()), this, TQ_SLOT(slotRemoveRatingCriteria()) ); + + TDEIconLoader il; + TDEPopupMenu *kpop = new TDEPopupMenu( criteriaListView ); + kpop->insertItem( il.loadIcon( "editshred", TDEIcon::NoGroup, 16 ), i18n( "&Delete" ), this, TQ_SLOT( slotRemoveRatingCriteria() ), Key_Delete ); + + for ( ElementList::const_iterator criteria_it = criteriaList.begin(); criteria_it != criteriaList.end(); ++criteria_it ) { + criteriaComboBox->insertItem( ( *criteria_it ).name ); + //criteriaComboBox->completionObject()->addItem( ( *criteria_it ).name ); + } + + ratingID = -1; +} + +/* + * Destroys the object and frees any allocated resources + */ +EditRatingDialog::~EditRatingDialog() +{ + // no need to delete child widgets, TQt does it all for us +} + +/* + * Sets the strings of the subwidgets using the current + * language. + */ +void EditRatingDialog::languageChange() +{ + criteriaLabel->setText( i18n( "Criteria:" ) ); + starsLabel->setText( i18n( "Stars:" ) ); + addButton->setText( i18n( "Add" ) ); + removeButton->setText( i18n( "&Delete" ) ); + criteriaListView->header()->setLabel( 0, i18n( "Criteria" ) ); + criteriaListView->header()->setLabel( 1, i18n( "Stars" ) ); + commentsLabel->setText( i18n( "Comments:" ) ); + raterLabel->setText( i18n( "Rater:" ) ); +} + +void EditRatingDialog::itemRenamed(TQListViewItem* it, const TQString &, int c) +{ + if ( c == 1 ) { + bool ok = false; + MixedNumber stars_mn = MixedNumber::fromString(it->text(c),&ok); + if ( ok && !it->text(c).isEmpty() ) { + double stars = TQMAX(0,TQMIN(stars_mn.toDouble(),5)); //force to between 0 and 5 + TQPixmap starsPic = Rating::starsPixmap( stars ); + it->setPixmap(c,starsPic); + it->setText(2,TQString::number(stars)); + } + else { + double stars = it->text(2).toDouble(); //col 2 holds the old value, which we'll set it back to + TQPixmap starsPic = Rating::starsPixmap( stars ); + it->setPixmap(c,starsPic); + } + + it->setText(c,TQString::null); + } +} + +Rating EditRatingDialog::rating() const +{ + Rating r; + + for ( TQListViewItem *it = criteriaListView->firstChild(); it; it = it->nextSibling() ) { + RatingCriteria rc; + rc.name = it->text(0); + rc.stars = it->text(2).toDouble(); + r.append( rc ); + } + + r.comment = commentsEdit->text(); + r.rater = raterEdit->text(); + + r.id = ratingID; + + return r; +} + +void EditRatingDialog::loadRating( const Rating &rating ) +{ + for ( RatingCriteriaList::const_iterator rc_it = rating.ratingCriteriaList.begin(); rc_it != rating.ratingCriteriaList.end(); ++rc_it ) { + addRatingCriteria(*rc_it); + } + + raterEdit->setText(rating.rater); + commentsEdit->setText(rating.comment); + + ratingID = rating.id; +} + +void EditRatingDialog::slotAddRatingCriteria() +{ + RatingCriteria r; + r.name = criteriaComboBox->lineEdit()->text().stripWhiteSpace(); + if ( r.name.isEmpty() ) + return; + + r.stars = starsWidget->text().toDouble(); + + addRatingCriteria(r); + + criteriaComboBox->lineEdit()->clear(); + starsWidget->clear(); + + criteriaComboBox->lineEdit()->setFocus(); +} + +void EditRatingDialog::addRatingCriteria( const RatingCriteria &rc ) +{ + TQListViewItem * it = new TQListViewItem(criteriaListView,rc.name); + + TQPixmap stars = Rating::starsPixmap(rc.stars); + if ( !stars.isNull() ) //there aren't zero stars + it->setPixmap(1,stars); + + it->setText(2,TQString::number(rc.stars)); +} + +void EditRatingDialog::slotRemoveRatingCriteria() +{ + delete criteriaListView->selectedItem(); +} + +#include "editratingdialog.moc" diff --git a/src/dialogs/editratingdialog.h b/src/dialogs/editratingdialog.h new file mode 100644 index 0000000..6996fb0 --- /dev/null +++ b/src/dialogs/editratingdialog.h @@ -0,0 +1,77 @@ +/*************************************************************************** +* Copyright (C) 2005 by Jason Kivlighn * +* [email protected] * +* * +* 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. * +***************************************************************************/ + +#ifndef EDITRATINGDIALOG_H +#define EDITRATINGDIALOG_H + +#include <kdialogbase.h> + +class TQVBoxLayout; +class TQHBoxLayout; +class TQGridLayout; +class TQSpacerItem; +class TQLabel; +class TQComboBox; +class KDoubleSpinBox; +class TQPushButton; +class TDEListView; +class TQListViewItem; +class TQTextEdit; +class TQLineEdit; +class RatingWidget; + +class Rating; +class RatingCriteria; +class ElementList; +class RatingCriteriaListView; + +class EditRatingDialog : public KDialogBase +{ +TQ_OBJECT + +public: + EditRatingDialog( const ElementList &criteriaList, const Rating &, TQWidget* parent = 0, const char* name = 0 ); + EditRatingDialog( const ElementList &criteriaList, TQWidget* parent = 0, const char* name = 0 ); + ~EditRatingDialog(); + + TQLabel* criteriaLabel; + TQComboBox* criteriaComboBox; + TQLabel* starsLabel; + RatingWidget *starsWidget; + TQPushButton* addButton; + TQPushButton* removeButton; + RatingCriteriaListView* criteriaListView; + TQLabel* commentsLabel; + TQTextEdit* commentsEdit; + TQLabel* raterLabel; + TQLineEdit* raterEdit; + + Rating rating() const; + +protected: + TQHBox* layout8; + TQHBox* layout2; + +protected slots: + virtual void languageChange(); + void slotAddRatingCriteria(); + void slotRemoveRatingCriteria(); + void itemRenamed(TQListViewItem* it, const TQString &, int c); + +private: + void init(const ElementList &criteriaList); + + void loadRating( const Rating & ); + void addRatingCriteria( const RatingCriteria &rc ); + + int ratingID; +}; + +#endif // EDITRATINGDIALOG_H diff --git a/src/dialogs/ingredientgroupsdialog.cpp b/src/dialogs/ingredientgroupsdialog.cpp new file mode 100644 index 0000000..dcfae3d --- /dev/null +++ b/src/dialogs/ingredientgroupsdialog.cpp @@ -0,0 +1,65 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "ingredientgroupsdialog.h" + +#include <tqpushbutton.h> +#include <tqlayout.h> +#include <tqvbox.h> + +#include <tdelocale.h> +#include <kdialog.h> + +#include "widgets/krelistview.h" +#include "widgets/headerlistview.h" + +IngredientGroupsDialog::IngredientGroupsDialog( RecipeDB *db, TQWidget *parent, const char *name ) : TQWidget(parent,name), database(db) +{ + TQHBoxLayout* layout = new TQHBoxLayout( this, KDialog::marginHint(), KDialog::spacingHint() ); + + headerListView = new KreListView ( this, i18n( "Header list" ), true, 0 ); + StdHeaderListView *list_view = new StdHeaderListView( headerListView, database, true ); + headerListView->setListView( list_view ); + headerListView->setSizePolicy( TQSizePolicy( TQSizePolicy::Ignored, TQSizePolicy::MinimumExpanding ) ); + layout->addWidget(headerListView); + + TQVBoxLayout *buttonLayout = new TQVBoxLayout(NULL, 0, KDialog::spacingHint()); + TQPushButton *addHeaderButton = new TQPushButton( this ); + addHeaderButton->setText( "+" ); + addHeaderButton->setMinimumSize( TQSize( 30, 30 ) ); + addHeaderButton->setMaximumSize( TQSize( 30, 30 ) ); + addHeaderButton->setSizePolicy( TQSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ) ); + addHeaderButton->setFlat( true ); + buttonLayout->addWidget(addHeaderButton); + + TQSpacerItem* spacer_buttons = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + buttonLayout->addItem( spacer_buttons ); + + TQPushButton *removeHeaderButton = new TQPushButton( this ); + removeHeaderButton->setText( "-" ); + removeHeaderButton->setMinimumSize( TQSize( 30, 30 ) ); + removeHeaderButton->setMaximumSize( TQSize( 30, 30 ) ); + removeHeaderButton->setSizePolicy( TQSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ) ); + removeHeaderButton->setFlat( true ); + buttonLayout->addWidget(removeHeaderButton); + + TQSpacerItem* spacer_below_buttons = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::MinimumExpanding ); + buttonLayout->addItem( spacer_below_buttons ); + + layout->addLayout(buttonLayout); + + connect( addHeaderButton, TQ_SIGNAL( clicked() ), list_view, TQ_SLOT( createNew() ) ); + connect( removeHeaderButton, TQ_SIGNAL( clicked() ), list_view, TQ_SLOT( remove() ) ); +} + +void IngredientGroupsDialog::reload( ReloadFlags flag ) +{ + ( ( StdHeaderListView* ) headerListView->listView() ) ->reload(flag); +} diff --git a/src/dialogs/ingredientgroupsdialog.h b/src/dialogs/ingredientgroupsdialog.h new file mode 100644 index 0000000..2b8f585 --- /dev/null +++ b/src/dialogs/ingredientgroupsdialog.h @@ -0,0 +1,33 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef INGREDIENTGROUPSDIALOG_H +#define INGREDIENTGROUPSDIALOG_H + +#include <tqwidget.h> + +#include "widgets/dblistviewbase.h" + +class KreListView; +class RecipeDB; + +class IngredientGroupsDialog : public TQWidget +{ +public: + IngredientGroupsDialog( RecipeDB *db, TQWidget *parent, const char *name ); + + void reload( ReloadFlags flag = Load ); + +private: + KreListView *headerListView; + RecipeDB *database; +}; + +#endif diff --git a/src/dialogs/ingredientmatcherdialog.cpp b/src/dialogs/ingredientmatcherdialog.cpp new file mode 100644 index 0000000..b994ec8 --- /dev/null +++ b/src/dialogs/ingredientmatcherdialog.cpp @@ -0,0 +1,353 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "ingredientmatcherdialog.h" + +#include "datablocks/recipelist.h" +#include "widgets/ingredientlistview.h" +#include "datablocks/elementlist.h" +#include "backends/recipedb.h" +#include "widgets/krelistview.h" +#include "widgets/unitcombobox.h" +#include "widgets/fractioninput.h" +#include "widgets/amountunitinput.h" +#include "datablocks/mixednumber.h" +#include "recipeactionshandler.h" + +#include <tqheader.h> +#include <tqpainter.h> +#include <tqstringlist.h> +#include <tqlayout.h> +#include <tqgroupbox.h> + +#include <tdeapplication.h> +#include <kcursor.h> +#include <kiconloader.h> +#include <tdelocale.h> +#include <knuminput.h> +#include <tdeconfig.h> +#include <tdeglobal.h> +#include <kdebug.h> +#include <kdialogbase.h> + +#include "profiling.h" + +IngredientMatcherDialog::IngredientMatcherDialog( TQWidget *parent, RecipeDB *db ) : TQWidget( parent ) +{ + // Initialize internal variables + database = db; + + //Design the dialog + + TQVBoxLayout *dialogLayout = new TQVBoxLayout( this, 11, 6 ); + + // Ingredient list + TQHBoxLayout *layout2 = new TQHBoxLayout( 0, 0, 6, "layout2" ); + + allIngListView = new KreListView( this, TQString::null, true, 0 ); + StdIngredientListView *list_view = new StdIngredientListView(allIngListView,database); + list_view->setSelectionMode( TQListView::Multi ); + allIngListView->setListView(list_view); + layout2->addWidget( allIngListView ); + + TQVBoxLayout *layout1 = new TQVBoxLayout( 0, 0, 6, "layout1" ); + + TDEIconLoader il; + + addButton = new TQPushButton( this, "addButton" ); + addButton->setIconSet( il.loadIconSet( "forward", TDEIcon::Small ) ); + addButton->setFixedSize( TQSize( 32, 32 ) ); + layout1->addWidget( addButton ); + + removeButton = new TQPushButton( this, "removeButton" ); + removeButton->setIconSet( il.loadIconSet( "back", TDEIcon::Small ) ); + removeButton->setFixedSize( TQSize( 32, 32 ) ); + layout1->addWidget( removeButton ); + TQSpacerItem *spacer1 = new TQSpacerItem( 51, 191, TQSizePolicy::Minimum, TQSizePolicy::Expanding ); + layout1->addItem( spacer1 ); + layout2->addLayout( layout1 ); + + ingListView = new KreListView( this, TQString::null, true ); + ingListView->listView() ->addColumn( i18n( "Ingredient (required?)" ) ); + ingListView->listView() ->addColumn( i18n( "Amount Available" ) ); + layout2->addWidget( ingListView ); + dialogLayout->addLayout( layout2 ); + + // Box to select allowed number of missing ingredients + missingBox = new TQHBox( this ); + missingNumberLabel = new TQLabel( missingBox ); + missingNumberLabel->setText( i18n( "Missing ingredients allowed:" ) ); + missingNumberSpinBox = new KIntSpinBox( missingBox ); + missingNumberSpinBox->setMinValue( -1 ); + missingNumberSpinBox->setSpecialValueText( i18n( "Any" ) ); + dialogLayout->addWidget(missingBox); + + // Found recipe list + recipeListView = new KreListView( this, i18n( "Matching Recipes" ), false, 1, missingBox ); + recipeListView->listView() ->setSizePolicy( TQSizePolicy::Ignored, TQSizePolicy::Ignored ); + recipeListView->listView() ->setAllColumnsShowFocus( true ); + + recipeListView->listView() ->addColumn( i18n( "Title" ) ); + + TDEConfig *config = TDEGlobal::config(); + config->setGroup( "Advanced" ); + bool show_id = config->readBoolEntry( "ShowID", false ); + recipeListView->listView() ->addColumn( i18n( "Id" ), show_id ? -1 : 0 ); + + recipeListView->listView() ->addColumn( i18n( "Missing Ingredients" ) ); + + recipeListView->listView() ->setSorting( -1 ); + dialogLayout->addWidget(recipeListView); + + RecipeActionsHandler *actionHandler = new RecipeActionsHandler( recipeListView->listView(), database, RecipeActionsHandler::Open | RecipeActionsHandler::Edit | RecipeActionsHandler::Export ); + + TQHBox *buttonBox = new TQHBox( this ); + + okButton = new TQPushButton( buttonBox ); + okButton->setIconSet( il.loadIconSet( "button_ok", TDEIcon::Small ) ); + okButton->setText( i18n( "Find matching recipes" ) ); + + //buttonBox->layout()->addItem( new TQSpacerItem( 10,10, TQSizePolicy::MinimumExpanding, TQSizePolicy::Fixed ) ); + + clearButton = new TQPushButton( buttonBox ); + clearButton->setIconSet( il.loadIconSet( "edit-clear", TDEIcon::Small ) ); + clearButton->setText( i18n( "Clear" ) ); + dialogLayout->addWidget(buttonBox); + + // Connect signals & slots + connect ( okButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( findRecipes() ) ); + connect ( clearButton, TQ_SIGNAL( clicked() ), recipeListView->listView(), TQ_SLOT( clear() ) ); + connect ( clearButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( unselectIngredients() ) ); + connect ( actionHandler, TQ_SIGNAL( recipeSelected( int, int ) ), TQ_SIGNAL( recipeSelected( int, int ) ) ); + connect( addButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( addIngredient() ) ); + connect( removeButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( removeIngredient() ) ); + connect( ingListView->listView(), TQ_SIGNAL( doubleClicked( TQListViewItem*, const TQPoint &, int ) ), TQ_SLOT( itemRenamed( TQListViewItem*, const TQPoint &, int ) ) ); +} + +IngredientMatcherDialog::~IngredientMatcherDialog() +{ +} + +void IngredientMatcherDialog::itemRenamed( TQListViewItem* item, const TQPoint &, int col ) +{ + if ( col == 1 ) { + KDialogBase amountEditDialog(this,"IngredientMatcherAmountEdit", + false, i18n("Enter amount"), KDialogBase::Cancel | KDialogBase::Ok, KDialogBase::Ok); + + TQGroupBox *box = new TQGroupBox( 1, Horizontal, i18n("Amount"), &amountEditDialog ); + AmountUnitInput *amountEdit = new AmountUnitInput( box, database ); + // Set the values from the item + if ( !item->text(1).isEmpty() ) { + amountEdit->setAmount( MixedNumber::fromString(item->text(2)) ); + Unit u; + u.id = item->text(3).toInt(); + amountEdit->setUnit( u ); + } else { + amountEdit->setAmount( -1 ); + Unit u; + u.id = -1; + amountEdit->setUnit( u ); + } + + amountEditDialog.setMainWidget(box); + + if ( amountEditDialog.exec() == TQDialog::Accepted ) { + MixedNumber amount = amountEdit->amount(); + Unit unit = amountEdit->unit(); + + if ( amount.toDouble() <= 1e-5 ) { + amount = -1; + unit.id = -1; + + item->setText(1,TQString::null); + } else { + item->setText(1,amount.toString()+" "+((amount.toDouble()>1)?unit.plural:unit.name)); + } + + item->setText(2,amount.toString()); + item->setText(3,TQString::number(unit.id)); + + IngredientList::iterator ing_it = m_item_ing_map[item]; + (*ing_it).amount = amount.toDouble(); + (*ing_it).units = unit; + } + } +} + +void IngredientMatcherDialog::addIngredient() +{ + TQPtrList<TQListViewItem> items = allIngListView->listView()->selectedItems(); + if ( items.count() > 0 ) { + TQPtrListIterator<TQListViewItem> it(items); + TQListViewItem *item; + while ( (item = it.current()) != 0 ) { + bool dup = false; + for ( TQListViewItem *exists_it = ingListView->listView()->firstChild(); exists_it; exists_it = exists_it->nextSibling() ) { + if ( exists_it->text( 0 ) == item->text( 0 ) ) { + dup = true; + break; + } + } + + if ( !dup ) { + TQListViewItem * new_item = new TQCheckListItem( ingListView->listView(), item->text( 0 ), TQCheckListItem::CheckBox ); + + ingListView->listView() ->setSelected( new_item, true ); + ingListView->listView() ->ensureItemVisible( new_item ); + allIngListView->listView() ->setSelected( item, false ); + + m_item_ing_map.insert( new_item, m_ingredientList.append( Ingredient( item->text( 0 ), 0, Unit(), -1, item->text( 1 ).toInt() ) ) ); + } + ++it; + } + } +} + +void IngredientMatcherDialog::removeIngredient() +{ + TQListViewItem * item = ingListView->listView() ->selectedItem(); + if ( item ) { + for ( IngredientList::iterator it = m_ingredientList.begin(); it != m_ingredientList.end(); ++it ) { + if ( *m_item_ing_map.find( item ) == it ) { + m_ingredientList.remove( it ); + m_item_ing_map.remove( item ); + break; + } + } + delete item; + } +} + +void IngredientMatcherDialog::unselectIngredients() +{ + ingListView->listView()->clear(); + for ( TQListViewItem *it = allIngListView->listView()->firstChild(); it; it = it->nextSibling() ) + allIngListView->listView()->setSelected(it,false); +} + +void IngredientMatcherDialog::findRecipes( void ) +{ + TDEApplication::setOverrideCursor( KCursor::waitCursor() ); + + START_TIMER("Ingredient Matcher: loading database data"); + + RecipeList rlist; + database->loadRecipes( &rlist, RecipeDB::Title | RecipeDB::NamesOnly | RecipeDB::Ingredients | RecipeDB::IngredientAmounts ); + + END_TIMER(); + START_TIMER("Ingredient Matcher: analyzing data for matching recipes"); + + // Clear the list + recipeListView->listView() ->clear(); + + // Now show the recipes with ingredients that are contained in the previous set + // of ingredients + RecipeList incompleteRecipes; + TQValueList <int> missingNumbers; + TQValueList <IngredientList> missingIngredients; + + RecipeList::Iterator it; + for ( it = rlist.begin();it != rlist.end();++it ) { + IngredientList il = ( *it ).ingList; + if ( il.isEmpty() ) + continue; + + IngredientList missing; + if ( m_ingredientList.containsSubSet( il, missing, true, database ) ) { + new CustomRecipeListItem( recipeListView->listView(), *it ); + } + else { + incompleteRecipes.append( *it ); + missingIngredients.append( missing ); + missingNumbers.append( missing.count() ); + } + } + END_TIMER(); + + //Check if the user wants to show missing ingredients + + if ( missingNumberSpinBox->value() == 0 ) { + TDEApplication::restoreOverrideCursor(); + return ; + } //"None" + + + + START_TIMER("Ingredient Matcher: searching for and displaying partial matches"); + + IngredientList requiredIngredients; + for ( TQListViewItem *it = ingListView->listView()->firstChild(); it; it = it->nextSibling() ) { + if ( ((TQCheckListItem*)it)->isOn() ) + requiredIngredients << *m_item_ing_map[it]; + } + + // Classify recipes with missing ingredients in different lists by ammount + TQValueList<int>::Iterator nit; + TQValueList<IngredientList>::Iterator ilit; + int missingNoAllowed = missingNumberSpinBox->value(); + + if ( missingNoAllowed == -1 ) // "Any" + { + for ( nit = missingNumbers.begin();nit != missingNumbers.end();++nit ) + if ( ( *nit ) > missingNoAllowed ) + missingNoAllowed = ( *nit ); + } + + + for ( int missingNo = 1; missingNo <= missingNoAllowed; missingNo++ ) { + nit = missingNumbers.begin(); + ilit = missingIngredients.begin(); + + bool titleShownYet = false; + + for ( it = incompleteRecipes.begin();it != incompleteRecipes.end();++it, ++nit, ++ilit ) { + if ( !( *it ).ingList.containsAny( m_ingredientList ) ) + continue; + + if ( !( *it ).ingList.containsSubSet( requiredIngredients ) ) + continue; + + if ( ( *nit ) == missingNo ) { + if ( !titleShownYet ) { + new SectionItem( recipeListView->listView(), i18n( "You are missing 1 ingredient for:", "You are missing %n ingredients for:", missingNo ) ); + titleShownYet = true; + } + new CustomRecipeListItem( recipeListView->listView(), *it, *ilit ); + } + } + } + END_TIMER(); + + TDEApplication::restoreOverrideCursor(); +} + +void IngredientMatcherDialog::reload( ReloadFlags flag ) +{ + ( ( StdIngredientListView* ) allIngListView->listView() ) ->reload(flag); +} + +void SectionItem::paintCell ( TQPainter * p, const TQColorGroup & /*cg*/, int column, int width, int /*align*/ ) +{ + // Draw the section's deco + p->setPen( TDEGlobalSettings::activeTitleColor() ); + p->setBrush( TDEGlobalSettings::activeTitleColor() ); + p->drawRect( 0, 0, width, height() ); + + // Draw the section's text + if ( column == 0 ) { + TQFont titleFont = TDEGlobalSettings::windowTitleFont (); + p->setFont( titleFont ); + + p->setPen( TDEGlobalSettings::activeTextColor() ); + p->drawText( 0, 0, width, height(), TQt::AlignLeft | TQt::AlignVCenter, mText ); + } +} + +#include "ingredientmatcherdialog.moc" diff --git a/src/dialogs/ingredientmatcherdialog.h b/src/dialogs/ingredientmatcherdialog.h new file mode 100644 index 0000000..2bd54b3 --- /dev/null +++ b/src/dialogs/ingredientmatcherdialog.h @@ -0,0 +1,156 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* * +* Copyright (C) 2006 Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef INGREDIENTMATCHERDIALOG_H +#define INGREDIENTMATCHERDIALOG_H + +#include "datablocks/element.h" +#include "datablocks/ingredientlist.h" +#include "datablocks/recipe.h" +#include "widgets/recipelistview.h" +#include "widgets/dblistviewbase.h" + +#include <tqfontmetrics.h> +#include <tqlabel.h> +#include <tqlistview.h> +#include <tqpushbutton.h> +#include <tqhbox.h> +#include <tqvbox.h> + +#include <kstringhandler.h> +#include <tdelocale.h> + +class KreListView; +class KIntSpinBox; +class RecipeDB; +class MixedNumber; + +/** +@author Unai Garro +*/ + +class CustomRecipeListItem : public RecipeListItem +{ +public: + CustomRecipeListItem( TQListView* qlv, const Recipe &r, const IngredientList &il ) : RecipeListItem( qlv, r ) + { + ingredientListStored = new TQStringList(); + IngredientList::ConstIterator ili; + for ( ili = il.begin();ili != il.end();++ili ) { + if ( (*ili).substitutes.count() > 0 ) { + TQStringList subs; + subs << ( *ili ).name; + for ( TQValueList<IngredientData>::const_iterator it = (*ili).substitutes.begin(); it != (*ili).substitutes.end(); ++it ) { + subs << (*it).name; + } + ingredientListStored->append( subs.join(TQString(" %1 ").arg(i18n("OR"))) ); + } + else + ingredientListStored->append( ( *ili ).name ); + } + + moveItem( qlv->lastItem() ); + } + CustomRecipeListItem( TQListView* qlv, const Recipe &r ) : RecipeListItem( qlv, r ) + { + ingredientListStored = 0; + + moveItem( qlv->lastItem() ); + } + + ~CustomRecipeListItem( void ) + { + delete ingredientListStored; + } + +private: + TQStringList *ingredientListStored; + +public: + virtual TQString text( int column ) const + { + if ( column == 2 && ingredientListStored ) + return ingredientListStored->join ( "," ); + else + return ( RecipeListItem::text( column ) ); + } +}; + +class SectionItem: public TQListViewItem +{ +public: + SectionItem( TQListView* qlv, TQString sectionText ) : TQListViewItem( qlv, qlv->lastItem() ) + { + mText = sectionText; + } + + ~SectionItem( void ) + {} + virtual void paintCell ( TQPainter * p, const TQColorGroup & cg, int column, int width, int align ); + +private: + TQString mText; + +public: + virtual TQString text( int column ) const + { + if ( column == 0 ) + return ( mText ); + else + return ( TQString::null ); + } +}; +class IngredientMatcherDialog: public TQWidget +{ + + TQ_OBJECT + +public: + + IngredientMatcherDialog( TQWidget *parent, RecipeDB* db ); + ~IngredientMatcherDialog(); + void reload( ReloadFlags flag = Load ); + +signals: + void recipeSelected( int, int ); + +private: + //Private variables + RecipeDB *database; + + //Widgets + + KreListView *allIngListView; + KreListView *ingListView; + + KreListView *recipeListView; + TQHBox *missingBox; + TQLabel *missingNumberLabel; + KIntSpinBox *missingNumberSpinBox; + + TQPushButton *okButton; + TQPushButton *clearButton; + TQPushButton *addButton; + TQPushButton *removeButton; + + IngredientList m_ingredientList; + TQMap<TQListViewItem*, IngredientList::iterator> m_item_ing_map; + +private slots: + void findRecipes( void ); + void unselectIngredients(); + void addIngredient(); + void removeIngredient(); + void itemRenamed( TQListViewItem*, const TQPoint &, int col ); +}; + +#endif diff --git a/src/dialogs/ingredientparserdialog.cpp b/src/dialogs/ingredientparserdialog.cpp new file mode 100644 index 0000000..0081bcf --- /dev/null +++ b/src/dialogs/ingredientparserdialog.cpp @@ -0,0 +1,299 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "ingredientparserdialog.h" + +#include <tqpushbutton.h> +#include <tqtextedit.h> +#include <tqlabel.h> +#include <tqlineedit.h> +#include <tqlayout.h> +#include <tqtooltip.h> +#include <tqwhatsthis.h> +#include <tqheader.h> +#include <tqapplication.h> +#include <tqclipboard.h> +#include <tqvbox.h> + +#include <tdelocale.h> +#include <kdebug.h> +#include <tdelistview.h> +#include <kpushbutton.h> +#include <tdemessagebox.h> +#include <tdeaction.h> +#include <tdepopupmenu.h> + +#include "datablocks/mixednumber.h" +#include "widgets/inglistviewitem.h" + +IngredientParserDialog::IngredientParserDialog( const UnitList &units, TQWidget* parent, const char* name ) + : KDialogBase( parent, name, true, i18n( "Ingredient Parser" ), + KDialogBase::Ok | KDialogBase::Cancel, KDialogBase::Ok ), + m_unitList(units) +{ + setButtonBoxOrientation( Vertical ); + + TQVBox *page = makeVBoxMainWidget(); + + textLabel1 = new TQLabel( page, "textLabel1" ); + textLabel1->setTextFormat( TQLabel::RichText ); + + ingredientTextEdit = new TQTextEdit( page, "ingredientTextEdit" ); + ingredientTextEdit->setTextFormat( TQTextEdit::PlainText ); + + parseButton = new KPushButton( page, "parseButton" ); + + previewLabel = new TQLabel( page, "previewLabel" ); + previewLabel->setTextFormat( TQLabel::RichText ); + + previewIngView = new TDEListView( page, "previewIngView" ); + previewIngView->setSorting(-1); + previewIngView->addColumn( i18n( "Ingredient" ) ); + previewIngView->addColumn( i18n( "Amount" ) ); + previewIngView->addColumn( i18n( "Unit" ) ); + previewIngView->addColumn( i18n( "Preparation Method" ) ); + + languageChange(); + setInitialSize( TQSize(577, 371).expandedTo(minimumSizeHint()) ); + + previewIngView->setItemsRenameable( true ); + previewIngView->setRenameable( 0, true ); + previewIngView->setRenameable( 1, true ); + previewIngView->setRenameable( 2, true ); + previewIngView->setRenameable( 3, true ); + + previewIngView->setSelectionMode( TQListView::Extended ); + + ingredientTextEdit->setText( TQApplication::clipboard()->text() ); + ingredientTextEdit->selectAll(); + + TQWidget *buttonWidget = new TQWidget( page ); + TQHBoxLayout *buttonBox = new TQHBoxLayout(buttonWidget); + TQSpacerItem *horizontalSpacing = new TQSpacerItem( 20, 20, TQSizePolicy::Expanding, TQSizePolicy::Minimum ); + buttonGroup = new TQPushButton( i18n("Set &Header"), buttonWidget ); + TQWhatsThis::add( buttonGroup, i18n("If an ingredient header is detected as an ingredient, select it and click this button so that Krecipes will recognize it as a header. All the ingredients below the header will be included within that group.\n\nAlternatively, if you select multiple ingredients and click this button, those ingredients will be grouped together.") ); + buttonBox->addWidget( buttonGroup ); + buttonBox->addItem( horizontalSpacing ); + + TDEPopupMenu *kpop = new TDEPopupMenu( previewIngView ); + kpop->insertItem( i18n( "&Delete" ), this, TQ_SLOT( removeIngredient() ), Key_Delete ); + kpop->insertItem( i18n("Set &Header") , this, TQ_SLOT( convertToHeader() ) ); + + connect( parseButton, TQ_SIGNAL(clicked()), this, TQ_SLOT(parseText()) ); + connect( buttonGroup, TQ_SIGNAL(clicked()), this, TQ_SLOT(convertToHeader()) ); +} + +IngredientParserDialog::~IngredientParserDialog() +{ + // no need to delete child widgets, TQt does it all for us +} + +void IngredientParserDialog::languageChange() +{ + textLabel1->setText( i18n( "To use: Paste a list of ingredient below, click \"Parse Text\", and then you may correct any incorrectly parsed ingredients.<br><b>Caution: Fields will be truncated if longer than the database allows</b>" ) ); + previewLabel->setText( i18n("Ingredients as understood by Krecipes:") ); + parseButton->setText( i18n( "Parse Text" ) ); + previewIngView->header()->setLabel( 0, i18n( "Ingredient" ) ); + previewIngView->header()->setLabel( 1, i18n( "Amount" ) ); + previewIngView->header()->setLabel( 2, i18n( "Unit" ) ); + previewIngView->header()->setLabel( 3, i18n( "Preparation Method" ) ); +} + +void IngredientParserDialog::accept() +{ + for ( TQListViewItem *it = previewIngView->firstChild(); it; it = it->nextSibling() ) { + if ( it->rtti() == INGGRPLISTVIEWITEM_RTTI ) { + TQString group = ((IngGrpListViewItem*)it)->group(); + for ( IngListViewItem *sub_it = (IngListViewItem*)it->firstChild(); sub_it; sub_it = (IngListViewItem*)sub_it->nextSibling() ) { + Ingredient ing = sub_it->ingredient(); + ing.group = group; + m_ingList.append(ing); + } + } + else + m_ingList.append(((IngListViewItem*)it)->ingredient()); + } + + TQDialog::accept(); +} + +void IngredientParserDialog::removeIngredient() +{ + delete previewIngView->selectedItem(); + if ( !previewIngView->firstChild() ) + enableButtonOK( false ); +} + +void IngredientParserDialog::convertToHeader() +{ + TQPtrList<TQListViewItem> items = previewIngView->selectedItems(); + if ( items.count() == 0 ) + return; + else if ( items.count() > 1 ) + convertToHeader(items); + else { //items.count = 1 + TQListViewItem *item = items.first(); + if ( item->rtti() == INGLISTVIEWITEM_RTTI ) { + TQListViewItem *new_item = new IngGrpListViewItem(previewIngView, + (item->parent())?item->parent():item, + ((IngListViewItem*)item)->ingredient().name, -1); + + TQListViewItem *next_sibling; + TQListViewItem *last_item = 0; + for ( TQListViewItem * it = (item->parent())?item->nextSibling():new_item->nextSibling(); it; it = next_sibling ) { + if ( it->rtti() == INGGRPLISTVIEWITEM_RTTI ) + break; + + next_sibling = it->nextSibling(); //get the next sibling of this item before we move it + + if ( it->parent() ) + it->parent()->takeItem(it); + else + previewIngView->takeItem( it ); + + new_item->insertItem( it ); + + if ( last_item ) + it->moveItem( last_item ); + last_item = it; + } + + new_item->setOpen(true); + + delete item; + } + } +} + +void IngredientParserDialog::convertToHeader( const TQPtrList<TQListViewItem> &items ) +{ + if ( items.count() > 0 ) { + TQPtrListIterator<TQListViewItem> it(items); + TQListViewItem *item = it.current(); + + if ( item->rtti() != INGLISTVIEWITEM_RTTI ) + return; + + TQString group = ((IngListViewItem*)item)->ingredient().name; + TQListViewItem *ingGroupItem = new IngGrpListViewItem(previewIngView, + (item->parent())?item->parent():item, group, -1); + delete item; //delete the ingredient header which was detected as an ingredient + ++it; + + TQListViewItem *last_item = 0; + while ( (item = it.current()) != 0 ) { + //ignore anything that isn't an ingredient (e.g. headers) + if ( item->rtti() == INGLISTVIEWITEM_RTTI ) { + if ( item->parent() ) + item->parent()->takeItem(item); + else + previewIngView->takeItem( item ); + + ingGroupItem->insertItem( item ); + + if ( last_item ) + item->moveItem( last_item ); + last_item = item; + } + + ++it; + } + + ingGroupItem->setOpen(true); + previewIngView->clearSelection(); + } +} + +void IngredientParserDialog::parseText() +{ + previewIngView->clear(); + + TQListViewItem *last_item = 0; + + int line_num = 0; + TQStringList ingredients = TQStringList::split("\n",ingredientTextEdit->text()); + for ( TQStringList::const_iterator it = ingredients.begin(); it != ingredients.end(); ++it ) { + TQString line = (*it).simplifyWhiteSpace(); + + ++line_num; + int format_at = 0; + Ingredient ing; + + + //======amount======// + int first_space = line.find( " ", format_at+1 ); + if ( first_space == -1 ) + first_space = line.length(); + + int second_space = line.find( " ", first_space+1 ); + if ( second_space == -1 ) + second_space = line.length(); + + Ingredient i; + bool ok; + i.setAmount(line.mid(format_at,second_space-format_at),&ok); + if ( !ok ) { + i.setAmount(line.mid(format_at,first_space-format_at),&ok); + if ( ok ) format_at = first_space+1; + } + else + format_at = second_space+1; + + if ( ok ) { + ing.amount = i.amount; + ing.amount_offset = i.amount_offset; + } + else + kdDebug()<<"no amount on line "<<line_num<<endl; + + + //======unit======// + first_space = line.find( " ", format_at+1 ); + if ( first_space == -1 ) + first_space = line.length(); + + bool isUnit = false; + TQString unitCheck = line.mid(format_at,first_space-format_at); + + for ( UnitList::const_iterator unit_it = m_unitList.begin(); unit_it != m_unitList.end(); ++unit_it ) { + if ( (*unit_it).name == unitCheck || (*unit_it).plural == unitCheck || (*unit_it).name_abbrev == unitCheck || (*unit_it).plural_abbrev == unitCheck ) { + isUnit = true; + format_at = first_space+1; + break; + } + } + + if ( isUnit ) { + ing.units.name = unitCheck; + ing.units.plural = unitCheck; + } + else + kdDebug()<<"no unit on line "<<line_num<<endl; + + + //======ingredient======// + int ing_end = line.find( TQRegExp("(,|;)"), format_at+1 ); + if ( ing_end == -1 ) + ing_end = line.length(); + + ing.name = line.mid(format_at,ing_end-format_at); + format_at = ing_end+2; + + + //======prep method======// + int end = line.length(); + ing.prepMethodList = ElementList::split(",",line.mid(format_at,end-format_at)); + + last_item = new IngListViewItem(previewIngView,last_item,ing); + enableButtonOK( true ); + } +} + +#include "ingredientparserdialog.moc" diff --git a/src/dialogs/ingredientparserdialog.h b/src/dialogs/ingredientparserdialog.h new file mode 100644 index 0000000..f9a6dbb --- /dev/null +++ b/src/dialogs/ingredientparserdialog.h @@ -0,0 +1,65 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef INGREDIENTPARSERDIALOG_H +#define INGREDIENTPARSERDIALOG_H + +#include <kdialogbase.h> + +#include "datablocks/ingredientlist.h" +#include "datablocks/unit.h" + +class TQVBoxLayout; +class TQHBoxLayout; +class TQGridLayout; +class TQSpacerItem; +class TQLabel; +class TQTextEdit; +class TQLineEdit; +class KPushButton; +class TDEListView; +class TQListViewItem; +class TQPushButton; + +class IngredientParserDialog : public KDialogBase +{ + TQ_OBJECT + +public: + IngredientParserDialog( const UnitList &units, TQWidget* parent = 0, const char* name = 0 ); + ~IngredientParserDialog(); + + IngredientList ingredients() const { return m_ingList; } + +protected: + TQLabel* textLabel1; + TQLabel *previewLabel; + TQTextEdit* ingredientTextEdit; + KPushButton* parseButton; + TDEListView* previewIngView; + TQPushButton* buttonGroup; + +protected slots: + virtual void accept(); + void parseText(); + void removeIngredient(); + + //Set Header button slot + virtual void convertToHeader(); + virtual void languageChange(); + +private: + void convertToHeader( const TQPtrList<TQListViewItem> &items ); + + UnitList m_unitList; + IngredientList m_ingList; +}; + +#endif // INGREDIENTPARSERDIALOG_H diff --git a/src/dialogs/ingredientsdialog.cpp b/src/dialogs/ingredientsdialog.cpp new file mode 100644 index 0000000..358e01f --- /dev/null +++ b/src/dialogs/ingredientsdialog.cpp @@ -0,0 +1,702 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "ingredientsdialog.h" +#include "backends/recipedb.h" +#include "createelementdialog.h" +#include "datablocks/ingredientpropertylist.h" +#include "datablocks/mixednumber.h" +#include "datablocks/weight.h" +#include "unitsdialog.h" +#include "usdadatadialog.h" +#include "selectpropertydialog.h" +#include "selectunitdialog.h" +#include "dependanciesdialog.h" +#include "widgets/ingredientlistview.h" +#include "widgets/weightinput.h" +#include "dialogs/ingredientgroupsdialog.h" +#include "dialogs/createingredientweightdialog.h" + +#include <tdeapplication.h> +#include <kcursor.h> +#include <kdebug.h> +#include <tdelocale.h> +#include <tdemessagebox.h> +#include <tdeglobal.h> +#include <tdeconfig.h> + +#include <tqheader.h> +#include <tqmessagebox.h> +#include <tqtabwidget.h> + +class WeightListItem : public TQListViewItem +{ +public: + WeightListItem( TQListView *listview, TQListViewItem *item, const Weight &w ) : TQListViewItem(listview,item), m_weight(w){} + + void setWeight( const Weight &w ) { m_weight = w; } + Weight weight() const { return m_weight; } + + void setAmountUnit( double amount, const Unit &unit, const Element &prepMethod ) + { + m_weight.perAmount = amount; + m_weight.perAmountUnitID = unit.id; + m_weight.perAmountUnit = (m_weight.perAmount>1)?unit.plural:unit.name; + m_weight.prepMethodID = prepMethod.id; + m_weight.prepMethod = prepMethod.name; + } + + void setWeightUnit( double weight, const Unit &unit ) + { + m_weight.weight = weight; + m_weight.weightUnitID = unit.id; + m_weight.weightUnit = (m_weight.weight>1)?unit.plural:unit.name; + } + + virtual TQString text( int c ) const + { + if ( c == 0 ) + return TQString::number(m_weight.weight)+" "+m_weight.weightUnit; + else if ( c == 1 ) + return TQString::number(m_weight.perAmount)+" " + +m_weight.perAmountUnit + +((m_weight.prepMethodID!=-1)?", "+m_weight.prepMethod:TQString::null); + else + return TQString::null; + } + +private: + Weight m_weight; +}; + +IngredientsDialog::IngredientsDialog( TQWidget* parent, RecipeDB *db ) : TQWidget( parent ) +{ + + // Store pointer to database + database = db; + + // Initialize internal variables + propertiesList = new IngredientPropertyList; + perUnitListBack = new ElementList; + + // Design dialog + + TQHBoxLayout* page_layout = new TQHBoxLayout( this, KDialog::marginHint(), KDialog::spacingHint() ); + + TQTabWidget *tabWidget = new TQTabWidget( this ); + + TQWidget *ingredientTab = new TQWidget( tabWidget ); + + layout = new TQGridLayout( ingredientTab, 1, 1, 0, 0 ); + TQSpacerItem* spacer_left = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::Minimum ); + layout->addItem( spacer_left, 1, 0 ); + TQSpacerItem* spacer_top = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( spacer_top, 0, 1 ); + + ingredientListView = new KreListView ( ingredientTab, i18n( "Ingredient list" ), true, 0 ); + StdIngredientListView *list_view = new StdIngredientListView( ingredientListView, database, true ); + ingredientListView->setListView( list_view ); + layout->addMultiCellWidget ( ingredientListView, 1, 5, 1, 1 ); + ingredientListView->setSizePolicy( TQSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::MinimumExpanding ) ); + + TQSpacerItem* spacer_rightIngredients = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::Minimum ); + layout->addItem( spacer_rightIngredients, 1, 2 ); + + + addIngredientButton = new TQPushButton( ingredientTab ); + addIngredientButton->setText( "+" ); + layout->addWidget( addIngredientButton, 1, 3 ); + addIngredientButton->setMinimumSize( TQSize( 30, 30 ) ); + addIngredientButton->setMaximumSize( TQSize( 30, 30 ) ); + addIngredientButton->setSizePolicy( TQSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ) ); + addIngredientButton->setFlat( true ); + + removeIngredientButton = new TQPushButton( ingredientTab ); + removeIngredientButton->setText( "-" ); + layout->addWidget( removeIngredientButton, 3, 3 ); + removeIngredientButton->setMinimumSize( TQSize( 30, 30 ) ); + removeIngredientButton->setMaximumSize( TQSize( 30, 30 ) ); + removeIngredientButton->setSizePolicy( TQSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ) ); + removeIngredientButton->setFlat( true ); + + TQSpacerItem* spacer_Ing_Buttons = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( spacer_Ing_Buttons, 2, 3 ); + + + TQSpacerItem* spacer_Ing_Units = new TQSpacerItem( 30, 5, TQSizePolicy::Fixed, TQSizePolicy::Minimum ); + layout->addItem( spacer_Ing_Units, 1, 4 ); + + + TDEConfig *config = TDEGlobal::config(); + config->setGroup( "Advanced" ); + bool show_id = config->readBoolEntry( "ShowID", false ); + + TQScrollView *scrollView1 = new TQScrollView( ingredientTab, "scrollView1" ); + scrollView1->enableClipper(true); + TQWidget *rightWidget = new TQWidget(scrollView1); + TQGridLayout *rightLayout = new TQGridLayout( rightWidget, 1, 1, 0, 0 ); + + unitsListView = new KreListView ( rightWidget, i18n( "Unit list" ) ); + unitsListView->listView() ->addColumn( i18n( "Units" ) ); + unitsListView->listView() ->addColumn( i18n( "Id" ), show_id ? -1 : 0 ); + unitsListView->listView() ->setSorting( 0 ); + unitsListView->listView() ->setAllColumnsShowFocus( true ); + rightLayout->addMultiCellWidget ( unitsListView, 1, 4, 0, 0 ); + unitsListView->listView() ->setMinimumWidth( 150 ); + unitsListView->setSizePolicy( TQSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::MinimumExpanding ) ); + + TQSpacerItem* spacer_rightUnits = new TQSpacerItem( 5, 5, TQSizePolicy::Fixed, TQSizePolicy::Minimum ); + rightLayout->addItem( spacer_rightUnits, 1, 1 ); + + addUnitButton = new TQPushButton( rightWidget ); + addUnitButton->setText( "+" ); + rightLayout->addWidget( addUnitButton, 1, 2 ); + addUnitButton->resize( TQSize( 30, 30 ) ); + addUnitButton->setMinimumSize( TQSize( 30, 30 ) ); + addUnitButton->setMaximumSize( TQSize( 30, 30 ) ); + addUnitButton->setSizePolicy( TQSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ) ); + addUnitButton->setFlat( true ); + + removeUnitButton = new TQPushButton( rightWidget ); + removeUnitButton->setText( "-" ); + rightLayout->addWidget( removeUnitButton, 3, 2 ); + removeUnitButton->resize( TQSize( 30, 30 ) ); + removeUnitButton->setMinimumSize( TQSize( 30, 30 ) ); + removeUnitButton->setMaximumSize( TQSize( 30, 30 ) ); + removeUnitButton->setSizePolicy( TQSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ) ); + removeUnitButton->setFlat( true ); + TQSpacerItem* spacer_Units_Properties = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + rightLayout->addItem( spacer_Units_Properties, 2, 2 ); + + + propertiesListView = new KreListView ( rightWidget, i18n( "Ingredient Properties" ) ); + rightLayout->addMultiCellWidget ( propertiesListView, 6, 9, 0, 0 ); + + propertiesListView->listView() ->addColumn( i18n( "Property" ) ); + propertiesListView->listView() ->addColumn( i18n( "Amount" ) ); + propertiesListView->listView() ->addColumn( i18n( "Units" ) ); + propertiesListView->listView() ->addColumn( i18n( "Id" ), show_id ? -1 : 0 ); + propertiesListView->listView() ->setAllColumnsShowFocus( true ); + propertiesListView->listView() ->setSizePolicy( TQSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::MinimumExpanding ) ); + propertiesListView->listView() ->setSorting( -1 ); // Disable sorting. For the moment, the order is important to identify the per_units ID corresponding to this row. So the user shouldn't change this order. + + addPropertyButton = new TQPushButton( rightWidget ); + addPropertyButton->setText( "+" ); + rightLayout->addWidget( addPropertyButton, 6, 2 ); + addPropertyButton->resize( TQSize( 30, 30 ) ); + addPropertyButton->setMinimumSize( TQSize( 30, 30 ) ); + addPropertyButton->setMaximumSize( TQSize( 30, 30 ) ); + addPropertyButton->setSizePolicy( TQSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ) ); + addPropertyButton->setFlat( true ); + + removePropertyButton = new TQPushButton( rightWidget ); + removePropertyButton->setText( "-" ); + rightLayout->addWidget( removePropertyButton, 8, 2 ); + removePropertyButton->resize( TQSize( 30, 30 ) ); + removePropertyButton->setMinimumSize( TQSize( 30, 30 ) ); + removePropertyButton->setMaximumSize( TQSize( 30, 30 ) ); + removePropertyButton->setSizePolicy( TQSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ) ); + removePropertyButton->setFlat( true ); + + TQSpacerItem* spacer_Prop_Buttons = new TQSpacerItem( 9, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + rightLayout->addItem( spacer_Prop_Buttons, 7, 2 ); + + weightsListView = new KreListView ( rightWidget, i18n( "Ingredient Weights" ) ); + weightsListView->listView() ->addColumn( i18n( "Weight" ) ); + weightsListView->listView() ->addColumn( i18n( "Per Amount" ) ); + weightsListView->listView() ->setAllColumnsShowFocus( true ); + rightLayout->addMultiCellWidget ( weightsListView, 10, 14, 0, 0 ); + weightsListView->listView() ->setMinimumWidth( 150 ); + weightsListView->setSizePolicy( TQSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::MinimumExpanding ) ); + + //TQSpacerItem* spacer_rightWeights = new TQSpacerItem( 5, 5, TQSizePolicy::Fixed, TQSizePolicy::Minimum ); + //layout->addItem( spacer_rightWeights, 1, 6 ); + + addWeightButton = new TQPushButton( rightWidget ); + addWeightButton->setText( "+" ); + rightLayout->addWidget( addWeightButton, 10, 2 ); + addWeightButton->resize( TQSize( 30, 30 ) ); + addWeightButton->setMinimumSize( TQSize( 30, 30 ) ); + addWeightButton->setMaximumSize( TQSize( 30, 30 ) ); + addWeightButton->setSizePolicy( TQSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ) ); + addWeightButton->setFlat( true ); + + removeWeightButton = new TQPushButton( rightWidget ); + removeWeightButton->setText( "-" ); + rightLayout->addWidget( removeWeightButton, 12, 2 ); + removeWeightButton->resize( TQSize( 30, 30 ) ); + removeWeightButton->setMinimumSize( TQSize( 30, 30 ) ); + removeWeightButton->setMaximumSize( TQSize( 30, 30 ) ); + removeWeightButton->setSizePolicy( TQSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ) ); + removeWeightButton->setFlat( true ); + + TQSpacerItem* spacer_Weight_Properties = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + rightLayout->addItem( spacer_Weight_Properties, 11, 2 ); + + TQSpacerItem* spacerBottom = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::Expanding ); + rightLayout->addItem( spacerBottom, 13, 2 ); + + scrollView1->setResizePolicy( TQScrollView::AutoOneFit ); + layout->addMultiCellWidget(scrollView1,1,4,5,5); + + TQPushButton *loadUsdaButton = new TQPushButton( ingredientTab ); + loadUsdaButton->setText( i18n( "Load USDA data" ) ); + loadUsdaButton->setFlat( true ); + layout->addWidget(loadUsdaButton,5,5); + + scrollView1->addChild(rightWidget); + scrollView1->setSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::Ignored ); + + + inputBox = new KDoubleNumInput( propertiesListView->listView() ->viewport() ); + propertiesListView->listView() ->addChild( inputBox ); + inputBox->hide(); + + tabWidget->insertTab( ingredientTab, i18n( "Ingredients" ) ); + + groupsDialog = new IngredientGroupsDialog(database,tabWidget,"groupsDialog"); + tabWidget->insertTab( groupsDialog, i18n( "Headers" ) ); + + page_layout->addWidget( tabWidget ); + + // Initialize + unitList = new UnitList; + + // Signals & Slots + connect( ingredientListView->listView(), TQ_SIGNAL( selectionChanged() ), this, TQ_SLOT( updateLists() ) ); + connect( addIngredientButton, TQ_SIGNAL( clicked() ), list_view, TQ_SLOT( createNew() ) ); + connect( addUnitButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( addUnitToIngredient() ) ); + connect( removeUnitButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( removeUnitFromIngredient() ) ); + connect( addWeightButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( addWeight() ) ); + connect( removeWeightButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( removeWeight() ) ); + connect( removeIngredientButton, TQ_SIGNAL( clicked() ), list_view, TQ_SLOT( remove + () ) ); + connect( addPropertyButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( addPropertyToIngredient() ) ); + connect( removePropertyButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( removePropertyFromIngredient() ) ); + connect( propertiesListView->listView(), TQ_SIGNAL( executed( TQListViewItem* ) ), this, TQ_SLOT( insertPropertyEditBox( TQListViewItem* ) ) ); + connect( propertiesListView->listView(), TQ_SIGNAL( selectionChanged() ), inputBox, TQ_SLOT( hide() ) ); + connect( inputBox, TQ_SIGNAL( valueChanged( double ) ), this, TQ_SLOT( setPropertyAmount( double ) ) ); + + connect( weightsListView->listView(), TQ_SIGNAL( doubleClicked( TQListViewItem*, const TQPoint &, int ) ), TQ_SLOT( itemRenamed( TQListViewItem*, const TQPoint &, int ) ) ); + + connect( loadUsdaButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( openUSDADialog() ) ); +} + + +IngredientsDialog::~IngredientsDialog() +{ + delete unitList; + delete perUnitListBack; + delete propertiesList; +} + +void IngredientsDialog::reloadIngredientList( ReloadFlags flag ) +{ + ( ( StdIngredientListView* ) ingredientListView->listView() ) ->reload(flag); + + // Reload Unit List + updateLists(); + +} + +void IngredientsDialog::reloadUnitList() +{ + + int ingredientID = -1; + // Find selected ingredient + TQListViewItem *it; + it = ingredientListView->listView() ->selectedItem(); + + if ( it ) { // Check if an ingredient is selected first + ingredientID = it->text( 1 ).toInt(); + } + + + unitList->clear(); + unitsListView->listView() ->clear(); + + if ( ingredientID >= 0 ) { + database->loadPossibleUnits( ingredientID, unitList ); + + //Populate this data into the TDEListView + + for ( UnitList::const_iterator unit_it = unitList->begin(); unit_it != unitList->end(); ++unit_it ) { + if ( !( *unit_it ).name.isEmpty() ) { + ( void ) new TQListViewItem( unitsListView->listView(), ( *unit_it ).name, TQString::number( ( *unit_it ).id ) ); + } + } + + // Select the first unit + unitsListView->listView() ->setSelected( unitsListView->listView() ->firstChild(), true ); + + } +} + +void IngredientsDialog::addWeight() +{ + TQListViewItem *it = ingredientListView->listView()->selectedItem(); + if ( it ) { + CreateIngredientWeightDialog weightDialog( this, database ); + if ( weightDialog.exec() == TQDialog::Accepted ) { + Weight w = weightDialog.weight(); + w.ingredientID = it->text( 1 ).toInt(); + database->addIngredientWeight( w ); + + TQListViewItem * lastElement = weightsListView->listView()->lastItem(); + + WeightListItem *weight_it = new WeightListItem( weightsListView->listView(), lastElement, w ); + weight_it->setAmountUnit( w.perAmount, database->unitName(w.perAmountUnitID), + Element(w.prepMethod,w.prepMethodID) + ); + weight_it->setWeightUnit( w.weight, database->unitName(w.weightUnitID) ); + } + } +} + +void IngredientsDialog::removeWeight() +{ + TQListViewItem *it = weightsListView->listView() ->selectedItem(); + if ( it ) { + switch ( KMessageBox::warningContinueCancel(this, i18n("Recipes may require this information for nutrient analysis. Are you sure you want to delete this entry?"), TQString::null, KStdGuiItem::cont(), "DeleteIngredientWeight") ) { + case KMessageBox::Continue: + database->removeIngredientWeight( ((WeightListItem*)it)->weight().id ); + delete it; + break; + default: break; + } + } +} + +void IngredientsDialog::itemRenamed( TQListViewItem* item, const TQPoint &, int col ) +{ + WeightListItem *weight_it = (WeightListItem*)item; + Weight w = weight_it->weight(); + + if ( col == 0 ) { + KDialogBase amountEditDialog(this,"WeightAmountEdit", + false, i18n("Enter amount"), KDialogBase::Cancel | KDialogBase::Ok, KDialogBase::Ok); + + TQGroupBox *box = new TQGroupBox( 1, Horizontal, i18n("Amount"), &amountEditDialog ); + AmountUnitInput *amountEdit = new AmountUnitInput( box, database, Unit::Mass, MixedNumber::DecimalFormat ); + + WeightListItem *it = (WeightListItem*)item; + Weight w = it->weight(); + + amountEdit->setAmount( w.weight ); + amountEdit->setUnit( Unit(w.weightUnit,w.weightUnit,w.weightUnitID) ); + + amountEditDialog.setMainWidget(box); + + if ( amountEditDialog.exec() == TQDialog::Accepted ) { + MixedNumber amount = amountEdit->amount(); + Unit unit = amountEdit->unit(); + + it->setWeightUnit( amount.toDouble(), unit ); + database->addIngredientWeight( it->weight() ); + } + } + else if ( col == 1 ) { + KDialogBase amountEditDialog(this,"PerAmountEdit", + false, i18n("Enter amount"), KDialogBase::Cancel | KDialogBase::Ok, KDialogBase::Ok); + + TQGroupBox *box = new TQGroupBox( 1, Horizontal, i18n("Amount"), &amountEditDialog ); + WeightInput *amountEdit = new WeightInput( box, database, Unit::All, MixedNumber::DecimalFormat ); + + WeightListItem *it = (WeightListItem*)item; + Weight w = it->weight(); + + amountEdit->setAmount( w.perAmount ); + amountEdit->setUnit( Unit(w.perAmountUnit,w.perAmountUnit,w.perAmountUnitID) ); + amountEdit->setPrepMethod( Element(w.prepMethod,w.prepMethodID) ); + + amountEditDialog.setMainWidget(box); + + if ( amountEditDialog.exec() == TQDialog::Accepted ) { + MixedNumber amount = amountEdit->amount(); + Unit unit = amountEdit->unit(); + + it->setAmountUnit( amount.toDouble(), unit, amountEdit->prepMethod() ); + database->addIngredientWeight( it->weight() ); + } + } +} + +void IngredientsDialog::addUnitToIngredient( void ) +{ + + // Find selected ingredient item + TQListViewItem * it; + int ingredientID = -1; + if ( ( it = ingredientListView->listView() ->selectedItem() ) ) { + ingredientID = it->text( 1 ).toInt(); + } + if ( ingredientID >= 0 ) // an ingredient was selected previously + { + UnitList allUnits; + database->loadUnits( &allUnits ); + + SelectUnitDialog unitsDialog( this, allUnits, SelectUnitDialog::HideEmptyUnit ); + + if ( unitsDialog.exec() == TQDialog::Accepted ) + { + int unitID = unitsDialog.unitID(); + + if ( !( database->ingredientContainsUnit( ingredientID, unitID ) ) ) + database->addUnitToIngredient( ingredientID, unitID ); // Add chosen unit to ingredient in database + else { + TQMessageBox::information( this, i18n( "Unit Exists" ), i18n( "The ingredient contains already the unit that you have chosen." ) ); + } + reloadUnitList(); // Reload the list from database + } + } +} + +void IngredientsDialog::removeUnitFromIngredient( void ) +{ + + // Find selected ingredient/unit item combination + TQListViewItem * it; + int ingredientID = -1, unitID = -1; + if ( ( it = ingredientListView->listView() ->selectedItem() ) ) + ingredientID = it->text( 1 ).toInt(); + if ( ( it = unitsListView->listView() ->selectedItem() ) ) + unitID = it->text( 1 ).toInt(); + + if ( ( ingredientID >= 0 ) && ( unitID >= 0 ) ) // an ingredient/unit combination was selected previously + { + ElementList dependingRecipes, dependingPropertiesInfo; + + database->findIngredientUnitDependancies( ingredientID, unitID, &dependingRecipes, &dependingPropertiesInfo ); + + TQValueList<ListInfo> lists; + if ( !dependingRecipes.isEmpty() ) { + ListInfo info; + info.list = dependingRecipes; + info.name = i18n("Recipes"); + lists << info; + } + if ( !dependingPropertiesInfo.isEmpty() ) { + ListInfo info; + info.list = dependingPropertiesInfo; + info.name = i18n("Properties"); + lists << info; + } + + if ( lists.isEmpty() ) + database->removeUnitFromIngredient( ingredientID, unitID ); + else + { // must warn! + DependanciesDialog warnDialog( this, lists ); + if ( !dependingRecipes.isEmpty() ) + warnDialog.setCustomWarning( i18n("You are about to permanantly delete recipes from your database.") ); + if ( warnDialog.exec() == TQDialog::Accepted ) + database->removeUnitFromIngredient( ingredientID, unitID ); + } + reloadUnitList(); // Reload the list from database + reloadPropertyList(); // Properties could have been removed if a unit is removed, so we need to reload. + } +} + +void IngredientsDialog:: reloadPropertyList( void ) +{ + propertiesList->clear(); + propertiesListView->listView() ->clear(); + perUnitListBack->clear(); + + inputBox->hide(); + + + //If none is selected, select first item + TQListViewItem *it; + it = ingredientListView->listView() ->selectedItem(); + + //Populate this data into the TDEListView + if ( it ) { // make sure that the ingredient list is not empty + + database->loadProperties( propertiesList, it->text( 1 ).toInt() ); // load the list for this ingredient + for ( IngredientPropertyList::const_iterator prop_it = propertiesList->begin(); prop_it != propertiesList->end(); ++prop_it ) { + TQListViewItem * lastElement = propertiesListView->listView() ->lastItem(); + //Insert property after the last one (it's important to keep the order in the case of the properties to be able to identify the per_units ID later on). + ( void ) new TQListViewItem( propertiesListView->listView(), lastElement, (*prop_it).name, TQString::number( (*prop_it).amount ), (*prop_it).units + TQString( "/" ) + (*prop_it).perUnit.name, TQString::number( (*prop_it).id ) ); + // Store the perUnits with the ID for using later + Element perUnitEl; + perUnitEl.id = (*prop_it).perUnit.id; + perUnitEl.name = (*prop_it).perUnit.name; + perUnitListBack->append( perUnitEl ); + + } + } +} + +void IngredientsDialog::reloadWeightList( void ) +{ + weightsListView->listView() ->clear(); + + //If none is selected, select first item + TQListViewItem *it = ingredientListView->listView() ->selectedItem(); + + //Populate this data into the TDEListView + if ( it ) { // make sure that the ingredient list is not empty + WeightList list = database->ingredientWeightUnits( it->text( 1 ).toInt() ); // load the list for this ingredient + for ( WeightList::const_iterator weight_it = list.begin(); weight_it != list.end(); ++weight_it ) { + TQListViewItem * lastElement = weightsListView->listView() ->lastItem(); + + Weight w = *weight_it; + WeightListItem *weight_it1 = new WeightListItem( weightsListView->listView(), lastElement, w ); + weight_it1->setAmountUnit( w.perAmount, + database->unitName(w.perAmountUnitID), + Element((w.prepMethodID==-1)?TQString::null:database->prepMethodName(w.prepMethodID),w.prepMethodID) + ); + weight_it1->setWeightUnit( w.weight, database->unitName(w.weightUnitID) ); + } + } +} + +void IngredientsDialog:: updateLists( void ) +{ + reloadUnitList(); + reloadPropertyList(); + reloadWeightList(); +} + +void IngredientsDialog::addPropertyToIngredient( void ) +{ + + // Find selected ingredient item + TQListViewItem * it; + int ingredientID = -1; + if ( ( it = ingredientListView->listView() ->selectedItem() ) ) { + ingredientID = it->text( 1 ).toInt(); + } + if ( ingredientID >= 0 ) // an ingredient was selected previously + { + IngredientPropertyList allProperties; + database->loadProperties( &allProperties ); + UnitList unitList; + database->loadPossibleUnits( ingredientID, &unitList ); + SelectPropertyDialog propertyDialog( this, &allProperties, &unitList, SelectPropertyDialog::HideEmptyUnit ); + + if ( propertyDialog.exec() == TQDialog::Accepted ) + { + + int propertyID = propertyDialog.propertyID(); + int perUnitsID = propertyDialog.perUnitsID(); + if ( !( database->ingredientContainsProperty( ingredientID, propertyID, perUnitsID ) ) ) { + if ( ( propertyID >= 0 ) && ( perUnitsID >= 0 ) ) // check if the property is not -1 ... (not selected) + database->addPropertyToIngredient( ingredientID, propertyID, 0, perUnitsID ); // Add result chosen property to ingredient in database, with amount 0 by default + } + else { + TQMessageBox::information( this, i18n( "Property Exists" ), i18n( "The property you tried to add already exists in the ingredient with the same per units." ) ); + } + reloadPropertyList(); // Reload the list from database + } + } +} + +void IngredientsDialog::removePropertyFromIngredient( void ) +{ + + // Find selected ingredient/property item combination + TQListViewItem * it; + int ingredientID = -1, propertyID = -1; + int perUnitsID = -1; + if ( ( it = ingredientListView->listView() ->selectedItem() ) ) + ingredientID = it->text( 1 ).toInt(); + if ( ( it = propertiesListView->listView() ->selectedItem() ) ) + propertyID = it->text( 3 ).toInt(); + if ( propertyID >= 0 ) + perUnitsID = perUnitListBack->getElement( findPropertyNo( it ) ).id ; + + if ( ( ingredientID >= 0 ) && ( propertyID >= 0 ) && ( perUnitsID >= 0 ) ) // an ingredient/property combination was selected previously + { + ElementList results; + database->removePropertyFromIngredient( ingredientID, propertyID, perUnitsID ); + + reloadPropertyList(); // Reload the list from database + + } +} + +void IngredientsDialog::insertPropertyEditBox( TQListViewItem* it ) +{ + TQRect r = propertiesListView->listView() ->header() ->sectionRect( 1 ); + + r.moveBy( 0, propertiesListView->listView() ->itemRect( it ).y() ); //Move down to the item, note that its height is same as header's right now. + + r.setHeight( it->height() ); // Set the item's height + + inputBox->setGeometry( r ); + + inputBox->setValue( it->text( 1 ).toDouble() ); + inputBox->show(); +} + +void IngredientsDialog::setPropertyAmount( double amount ) +{ + TQListViewItem *ing_it = ingredientListView->listView() ->selectedItem(); // Find selected ingredient + TQListViewItem *prop_it = propertiesListView->listView() ->selectedItem(); + + if ( ing_it && prop_it ) // Appart from property, Check if an ingredient is selected first, just in case + { + prop_it->setText( 1, TQString::number( amount ) ); + int propertyID = prop_it->text( 3 ).toInt(); + int ingredientID = ing_it->text( 1 ).toInt(); + int per_units = perUnitListBack->getElement( findPropertyNo( prop_it ) ).id ; + database->changePropertyAmountToIngredient( ingredientID, propertyID, amount, per_units ); + } +} + +int IngredientsDialog::findPropertyNo( TQListViewItem * /*it*/ ) +{ + bool found = false; + int i = 0; + TQListViewItem* item = propertiesListView->listView() ->firstChild(); + while ( i < propertiesListView->listView() ->childCount() && !found ) { + if ( item == propertiesListView->listView() ->currentItem() ) + found = true; + else { + item = item->nextSibling(); + ++i; + } + } + if ( found ) { + return ( i ); + } + else { + return ( -1 ); + } +} + +void IngredientsDialog::reload( ReloadFlags flag ) +{ + reloadIngredientList( flag ); + groupsDialog->reload( flag ); +} + +void IngredientsDialog::openUSDADialog( void ) +{ + TQListViewItem * ing_it = ingredientListView->listView() ->selectedItem(); // Find selected ingredient + if ( ing_it ) { + TDEApplication::setOverrideCursor( KCursor::waitCursor() ); + USDADataDialog usda_dialog( Element( ing_it->text( 0 ), ing_it->text( 1 ).toInt() ), database, this ); + TDEApplication::restoreOverrideCursor(); + + if ( usda_dialog.exec() == TQDialog::Accepted ) { + reloadPropertyList(); //update property list upon success + reloadWeightList(); + } + } + else + TQMessageBox::information( this, TQString::null, i18n( "No ingredient selected." ) ); +} + +#include "ingredientsdialog.moc" diff --git a/src/dialogs/ingredientsdialog.h b/src/dialogs/ingredientsdialog.h new file mode 100644 index 0000000..32af993 --- /dev/null +++ b/src/dialogs/ingredientsdialog.h @@ -0,0 +1,93 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef INGREDIENTSDIALOG_H +#define INGREDIENTSDIALOG_H + +#include <tqpushbutton.h> +#include <tqvbox.h> +#include <tqlayout.h> +#include <tqwidget.h> +#include <knuminput.h> + +#include "widgets/krelistview.h" +#include "widgets/dblistviewbase.h" +#include "datablocks/unit.h" + +class KDoubleNumInput; + +class RecipeDB; +class ElementList; +class SelectUnitDialog; +class CreateElementDialog; +class IngredientPropertyList; +class SelectPropertyDialog; +class UnitsDialog; +class IngredientGroupsDialog; +class MixedNumber; + +class IngredientsDialog: public TQWidget +{ +TQ_OBJECT + +public: + IngredientsDialog( TQWidget* parent, RecipeDB *db ); + ~IngredientsDialog(); + void reload( ReloadFlags flag = Load ); + +private: + // Widgets + TQGridLayout* layout; + TQPushButton* addIngredientButton; + TQPushButton* removeIngredientButton; + TQPushButton* addUnitButton; + TQPushButton* removeUnitButton; + TQPushButton* addPropertyButton; + TQPushButton* removePropertyButton; + TQPushButton* addWeightButton; + TQPushButton* removeWeightButton; + KreListView* ingredientListView; + KreListView* unitsListView; + KreListView* propertiesListView; + KreListView* weightsListView; + TQPushButton* pushButton5; + KDoubleNumInput* inputBox; + IngredientGroupsDialog *groupsDialog; + + // Internal Methods + void reloadIngredientList( ReloadFlags flag = Load ); + void reloadUnitList( void ); + void reloadPropertyList( void ); + void reloadWeightList( void ); + int findPropertyNo( TQListViewItem *it ); + + // Internal Variables + RecipeDB *database; + UnitList *unitList; + IngredientPropertyList *propertiesList; + ElementList *perUnitListBack; + +private slots: + void addUnitToIngredient( void ); + void removeUnitFromIngredient( void ); + void addWeight(); + void removeWeight(); + void updateLists( void ); + void addPropertyToIngredient( void ); + void removePropertyFromIngredient( void ); + void insertPropertyEditBox( TQListViewItem* it ); + void setPropertyAmount( double amount ); + void openUSDADialog( void ); + void itemRenamed( TQListViewItem*, const TQPoint &, int col ); +}; + +#endif diff --git a/src/dialogs/pagesetupdialog.cpp b/src/dialogs/pagesetupdialog.cpp new file mode 100644 index 0000000..08e3fda --- /dev/null +++ b/src/dialogs/pagesetupdialog.cpp @@ -0,0 +1,310 @@ +/*************************************************************************** +* Copyright (C) 2003-2005 by Jason Kivlighn * +* ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "pagesetupdialog.h" + +#include <tqdir.h> +#include <tqlayout.h> +#include <tqhbox.h> +#include <tqfileinfo.h> +#include <tqpushbutton.h> +#include <tqpopupmenu.h> +#include <tqtooltip.h> +#include <tqtabwidget.h> +#include <tqlabel.h> + +#include <tdehtmlview.h> +#include <tdeapplication.h> +#include <kdebug.h> +#include <tdefiledialog.h> +#include <tdelocale.h> +#include <kiconloader.h> +#include <tdemessagebox.h> +#include <kstandarddirs.h> +#include <tdeaction.h> +#include <tdeconfig.h> +#include <kstdaction.h> +#include <tdetoolbar.h> +#include <tdepopupmenu.h> + +#include "setupdisplay.h" + +PageSetupDialog::PageSetupDialog( TQWidget *parent, const Recipe &sample, const TQString &configEntry ) : KDialog( parent, 0, true ), m_configEntry(configEntry) +{ + TDEIconLoader il; + + TQVBoxLayout * layout = new TQVBoxLayout( this ); + + TDEToolBar *toolbar = new TDEToolBar( this ); + TDEActionCollection *actionCollection = new TDEActionCollection( this ); + + TDEAction *std_open = KStdAction::open( 0, 0, 0 ); //use this to create a custom open action + TDEToolBarPopupAction *custom_open = new TDEToolBarPopupAction( std_open->text(), std_open->icon(), std_open->shortcut(), this, TQ_SLOT( loadFile() ), actionCollection, "open_popup" ); + + TDEPopupMenu *open_popup = custom_open->popupMenu(); + + open_popup->insertTitle( i18n( "Styles" ) ); + TQDir included_layouts( getIncludedLayoutDir(), "*.klo", TQDir::Name | TQDir::IgnoreCase, TQDir::Files ); + for ( unsigned int i = 0; i < included_layouts.count(); i++ ) { + int id = open_popup->insertItem( included_layouts[ i ].left(included_layouts[ i ].find(".")), this, TQ_SLOT( loadLayout( int ) ) ); + included_layouts_map.insert( id, included_layouts[ i ] ); + } + + open_popup->insertTitle( i18n( "Templates" ) ); + TQDir included_templates( getIncludedLayoutDir(), "*.template", TQDir::Name | TQDir::IgnoreCase, TQDir::Files ); + for ( unsigned int i = 0; i < included_templates.count(); i++ ) { + int id = open_popup->insertItem( included_templates[ i ].left(included_templates[ i ].find(".")).replace("_"," "), this, TQ_SLOT( loadTemplate( int ) ) ); + included_layouts_map.insert( id, included_templates[ i ] ); + } + + custom_open->plug( toolbar ); + + KStdAction::save( this, TQ_SLOT( saveLayout() ), actionCollection ) ->plug( toolbar ); + KStdAction::saveAs( this, TQ_SLOT( saveAsLayout() ), actionCollection ) ->plug( toolbar ); + KStdAction::redisplay( this, TQ_SLOT( reloadLayout() ), actionCollection ) ->plug( toolbar ); + + TDEToolBarPopupAction *shown_items = new TDEToolBarPopupAction( i18n( "Items Shown" ), "frame_edit" ); + shown_items->setDelayed( false ); + shown_items_popup = shown_items->popupMenu(); + shown_items_popup->insertTitle( i18n( "Show Items" ) ); + shown_items->plug( toolbar ); + layout->addWidget( toolbar ); + + TQLabel *help = new TQLabel(i18n("<i>Usage: Right-click any element to edit the look of that element.</i>"),this); + layout->addWidget( help ); + + m_htmlPart = new SetupDisplay(sample, this); + layout->addWidget( m_htmlPart->view() ); + + TQHBox *buttonsBox = new TQHBox( this ); + TQPushButton *okButton = new TQPushButton( il.loadIconSet( "ok", TDEIcon::Small ), i18n( "Save and Close" ), buttonsBox ); + TQPushButton *cancelButton = new TQPushButton( il.loadIconSet( "cancel", TDEIcon::Small ), i18n( "&Cancel" ), buttonsBox ); + layout->addWidget( buttonsBox ); + + connect( m_htmlPart, TQ_SIGNAL(itemVisibilityChanged(KreDisplayItem*,bool)), this, TQ_SLOT(updateItemVisibility(KreDisplayItem*,bool)) ); + connect( okButton, TQ_SIGNAL( clicked() ), TQ_SLOT( accept() ) ); + connect( cancelButton, TQ_SIGNAL( clicked() ), TQ_SLOT( reject() ) ); + + TDEConfig *config = TDEGlobal::config(); + config->setGroup( "Page Setup" ); + TQSize defaultSize(300,400); + resize(config->readSizeEntry( "WindowSize", &defaultSize )); + + //let's do everything we can to be sure at least some layout is loaded + TQString layoutFile = config->readEntry( m_configEntry+"Layout", locate( "appdata", "layouts/Default.klo" ) ); + if ( layoutFile.isEmpty() || !TQFile::exists( layoutFile ) ) + layoutFile = locate( "appdata", "layouts/Default.klo" ); + + TQString tmpl = config->readEntry( m_configEntry+"Template", locate( "appdata", "layouts/Default.template" ) ); + if ( tmpl.isEmpty() || !TQFile::exists( tmpl ) ) + tmpl = locate( "appdata", "layouts/Default.template" ); + kdDebug()<<"tmpl: "<<tmpl<<endl; + active_template = tmpl; + loadLayout( layoutFile ); + + initShownItems(); +} + +void PageSetupDialog::accept() +{ + if ( m_htmlPart->hasChanges() ) + saveLayout(); + + if ( !active_filename.isEmpty() ) { + TDEConfig * config = kapp->config(); + config->setGroup( "Page Setup" ); + config->writeEntry( m_configEntry+"Layout", active_filename ); + } + + if ( !active_template.isEmpty() ) { + TDEConfig * config = kapp->config(); + config->setGroup( "Page Setup" ); + config->writeEntry( m_configEntry+"Template", active_template ); + } + + TDEConfig *config = kapp->config(); + config->setGroup( "Page Setup" ); + config->writeEntry( "WindowSize", size() ); + + TQDialog::accept(); +} + +void PageSetupDialog::reject() +{ + if ( m_htmlPart->hasChanges() ) { + switch ( KMessageBox::questionYesNoCancel( this, i18n( "The recipe view layout has been modified.\nDo you want to save it?" ), i18n( "Save Layout?" ) ) ) { + case KMessageBox::Yes: + saveLayout(); + break; + case KMessageBox::No: + break; + default: + return ; + } + } + + TQDialog::reject(); +} + +void PageSetupDialog::updateItemVisibility( KreDisplayItem *item, bool visible ) +{ + shown_items_popup->setItemChecked( widget_popup_map[ item ], visible ); +} + +void PageSetupDialog::initShownItems() +{ + shown_items_popup->clear(); + + PropertiesMap properties = m_htmlPart->properties(); + + TQValueList<TQString> nameList; + TQMap<TQString,KreDisplayItem*> nameMap; + + for ( PropertiesMap::const_iterator it = properties.begin(); it != properties.end(); ++it ) { + nameList << it.key()->name; + nameMap.insert( it.key()->name, it.key() ); + } + qHeapSort( nameList ); + + for ( TQValueList<TQString>::const_iterator it = nameList.begin(); it != nameList.end(); ++it ) { + KreDisplayItem *item = nameMap[*it]; + if ( properties[item] & SetupDisplay::Visibility ) { + int new_id = shown_items_popup->insertItem ( *it ); + shown_items_popup->setItemChecked( new_id, item->show ); + shown_items_popup->connectItem( new_id, this, TQ_SLOT( setItemShown( int ) ) ); + + popup_widget_map.insert( new_id, item ); + widget_popup_map.insert( item, new_id ); + } + } +} + +void PageSetupDialog::setItemShown( int id ) +{ + shown_items_popup->setItemChecked( id, !shown_items_popup->isItemChecked( id ) ); + m_htmlPart->setItemShown( popup_widget_map[ id ], shown_items_popup->isItemChecked( id ) ); +} + +void PageSetupDialog::loadFile() +{ + TQString file = KFileDialog::getOpenFileName( locateLocal( "appdata", "layouts/" ), TQString("*.klo *.template|%1").arg(i18n("Krecipes style or template file")), this, i18n( "Select Layout" ) ); + + if ( file.endsWith(".klo") ) + loadLayout( file ); + else { + active_template = file; + m_htmlPart->loadTemplate( file ); + } +} + +void PageSetupDialog::loadTemplate( int popup_param ) +{ + active_template = getIncludedLayoutDir() + "/" + included_layouts_map[ popup_param ]; + m_htmlPart->loadTemplate( active_template ); +} + +void PageSetupDialog::loadLayout( int popup_param ) +{ + loadLayout( getIncludedLayoutDir() + "/" + included_layouts_map[ popup_param ] ); +} + +void PageSetupDialog::loadLayout( const TQString &filename ) +{ + if ( m_htmlPart->hasChanges() ) { + switch ( KMessageBox::questionYesNoCancel( this, i18n( "This layout has been modified.\nDo you want to save it?" ), i18n( "Save Layout?" ) ) ) { + case KMessageBox::Yes: + saveLayout(); + break; + case KMessageBox::No: + break; + default: + return ; + } + } + + if ( !filename.isEmpty() ) { + m_htmlPart->loadLayout( filename ); + setActiveFile( filename ); + } +} + +void PageSetupDialog::reloadLayout() +{ + m_htmlPart->reload(); +} + +void PageSetupDialog::saveLayout() +{ + if ( m_htmlPart->hasChanges() ) { + if ( have_write_perm ) + m_htmlPart->saveLayout( active_filename ); + else { + switch ( KMessageBox::warningYesNo( this, i18n( "Unable to save the layout because you do not have sufficient permissions to modify this file.\nWould you like to instead save the current layout to a new file?" ) ) ) { + case KMessageBox::Yes: + saveAsLayout(); + break; + default: + break; + } + } + } +} + +void PageSetupDialog::saveAsLayout() +{ + TQString filename = KFileDialog::getSaveFileName( TDEGlobal::instance() ->dirs() ->saveLocation( "appdata", "layouts/" ), "*.klo|Krecipes Layout (*.klo)", this, TQString::null ); + + if ( !filename.isEmpty() ) { + if ( haveWritePerm( filename ) ) { + m_htmlPart->saveLayout( filename ); + setActiveFile(filename); + } + else { + switch ( KMessageBox::warningYesNo( this, i18n( "You have selected a file that you do not have the permissions to write to.\nWould you like to select another file?" ) ) ) { + case KMessageBox::Yes: + saveAsLayout(); + break; + default: + break; + } + } + } +} + +TQString PageSetupDialog::getIncludedLayoutDir() const +{ + TQFileInfo file_info( locate( "appdata", "layouts/Default.klo" ) ); + return file_info.dirPath( true ); +} + +void PageSetupDialog::setActiveFile( const TQString &filename ) +{ + active_filename = filename; + have_write_perm = haveWritePerm( filename ); +} + +bool PageSetupDialog::haveWritePerm( const TQString &filename ) +{ + TQFileInfo info( filename ); + + if ( info.exists() ) //check that we can write to this particular file + { + TQFileInfo info( filename ); + return info.isWritable(); + } + else //check that we can write to the directory since the file doesn't exist + { + TQFileInfo dir_info( info.dirPath( true ) ); + return dir_info.isWritable(); + } +} + + +#include "pagesetupdialog.moc" diff --git a/src/dialogs/pagesetupdialog.h b/src/dialogs/pagesetupdialog.h new file mode 100644 index 0000000..70c2b4d --- /dev/null +++ b/src/dialogs/pagesetupdialog.h @@ -0,0 +1,72 @@ +/*************************************************************************** +* Copyright (C) 2003-2005 by Jason Kivlighn * +* ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef PAGESETUPDIALOG_H +#define PAGESETUPDIALOG_H + +#include <tqmap.h> + +#include <kdialog.h> + +#include "datablocks/recipe.h" + +class TDEPopupMenu; + +class SetupDisplay; +class KreDisplayItem; + +/** + * @author Jason Kivlighn + */ +class PageSetupDialog : public KDialog +{ + TQ_OBJECT + +public: + PageSetupDialog( TQWidget *parent, const Recipe &sample, const TQString &configEntry = TQString::null ); + ~PageSetupDialog() + {} + +protected: + virtual void accept(); + virtual void reject(); + void save(); + +private slots: + void loadFile(); + void loadLayout( int ); + void loadLayout( const TQString &filename ); + void loadTemplate( int ); + void reloadLayout(); + void saveLayout(); + void saveAsLayout(); + void updateItemVisibility( KreDisplayItem*, bool ); + void setItemShown( int id ); + +private: + TQString getIncludedLayoutDir() const; + void setActiveFile( const TQString &filename ); + bool haveWritePerm( const TQString &filename ); + void initShownItems(); + + SetupDisplay *m_htmlPart; + + TQString active_filename; + TQString active_template; + bool have_write_perm; + TQString m_configEntry; + + TQMap<int, KreDisplayItem*> popup_widget_map; + TQMap<KreDisplayItem*, int> widget_popup_map; + TQMap<int, TQString> included_layouts_map; + TDEPopupMenu *shown_items_popup; +}; + +#endif //PAGESETUPDIALOG_H diff --git a/src/dialogs/prepmethodsdialog.cpp b/src/dialogs/prepmethodsdialog.cpp new file mode 100644 index 0000000..066c09f --- /dev/null +++ b/src/dialogs/prepmethodsdialog.cpp @@ -0,0 +1,67 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "prepmethodsdialog.h" +#include "createelementdialog.h" +#include "dependanciesdialog.h" +#include "backends/recipedb.h" +#include "widgets/prepmethodlistview.h" + +#include <kdialog.h> +#include <tdelocale.h> +#include <tdemessagebox.h> + +PrepMethodsDialog::PrepMethodsDialog( TQWidget* parent, RecipeDB *db ) : TQWidget( parent ) +{ + + // Store pointer to database + database = db; + + TQHBoxLayout* layout = new TQHBoxLayout( this, KDialog::marginHint(), KDialog::spacingHint() ); + + //PrepMethod List + prepMethodListView = new StdPrepMethodListView( this, database, true ); + layout->addWidget( prepMethodListView ); + + //Buttons + TQVBoxLayout* vboxl = new TQVBoxLayout( KDialog::spacingHint() ); + + newPrepMethodButton = new TQPushButton( this ); + newPrepMethodButton->setText( i18n( "Create ..." ) ); + newPrepMethodButton->setFlat( true ); + vboxl->addWidget( newPrepMethodButton ); + + removePrepMethodButton = new TQPushButton( this ); + removePrepMethodButton->setText( i18n( "Delete" ) ); + removePrepMethodButton->setFlat( true ); + vboxl->addWidget( removePrepMethodButton ); + vboxl->addStretch(); + + layout->addLayout( vboxl ); + + //Connect Signals & Slots + + connect ( newPrepMethodButton, TQ_SIGNAL( clicked() ), prepMethodListView, TQ_SLOT( createNew() ) ); + connect ( removePrepMethodButton, TQ_SIGNAL( clicked() ), prepMethodListView, TQ_SLOT( remove + () ) ); +} + +PrepMethodsDialog::~PrepMethodsDialog() +{} + +// (Re)loads the data from the database +void PrepMethodsDialog::reload( ReloadFlags flag ) +{ + prepMethodListView->reload( flag ); +} + +#include "prepmethodsdialog.moc" diff --git a/src/dialogs/prepmethodsdialog.h b/src/dialogs/prepmethodsdialog.h new file mode 100644 index 0000000..5dbb01c --- /dev/null +++ b/src/dialogs/prepmethodsdialog.h @@ -0,0 +1,53 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef PREPMETHODSDIALOG_H +#define PREPMETHODSDIALOG_H + +#include <tqwidget.h> +#include <tqpushbutton.h> +#include <tqhbox.h> +#include <tqlayout.h> +#include <kiconloader.h> +#include <tdelistview.h> + +#include "widgets/dblistviewbase.h" + +class RecipeDB; +class StdPrepMethodListView; + +/** +@author Unai Garro +*/ + +class PrepMethodsDialog: public TQWidget +{ + + TQ_OBJECT + +public: + + PrepMethodsDialog( TQWidget* parent, RecipeDB *db ); + ~PrepMethodsDialog(); + void reload( ReloadFlags flag = Load ); +private: + // Internal data + RecipeDB *database; + //Widgets + TQGridLayout *layout; + StdPrepMethodListView *prepMethodListView; + TQHBox *buttonBar; + TQPushButton *newPrepMethodButton; + TQPushButton *removePrepMethodButton; + TDEIconLoader *il; +}; +#endif diff --git a/src/dialogs/propertiesdialog.cpp b/src/dialogs/propertiesdialog.cpp new file mode 100644 index 0000000..366eb63 --- /dev/null +++ b/src/dialogs/propertiesdialog.cpp @@ -0,0 +1,77 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "propertiesdialog.h" +#include <tdelocale.h> +#include <kdialog.h> +#include <tdemessagebox.h> +#include <tdeconfig.h> + +#include "backends/recipedb.h" +#include "createpropertydialog.h" +#include "widgets/propertylistview.h" + +PropertiesDialog::PropertiesDialog( TQWidget *parent, RecipeDB *db ) : TQWidget( parent ) +{ + + // Store pointer to database + database = db; + + // Design dialog + + TQHBoxLayout* layout = new TQHBoxLayout( this, KDialog::marginHint(), KDialog::spacingHint() ); + + propertyListView = new CheckPropertyListView( this, database, true ); + propertyListView->reload(); + + TDEConfig *config = TDEGlobal::config(); + config->setGroup("Formatting"); + TQStringList hiddenList = config->readListEntry("HiddenProperties"); + for ( TQCheckListItem *item = (TQCheckListItem*)propertyListView->firstChild(); item; item = (TQCheckListItem*)item->nextSibling() ) { + if ( !hiddenList.contains(item->text(0)) ) + item->setOn(true); + } + + layout->addWidget ( propertyListView ); + + TQVBoxLayout* vboxl = new TQVBoxLayout( KDialog::spacingHint() ); + addPropertyButton = new TQPushButton( this ); + addPropertyButton->setText( i18n( "Create ..." ) ); + addPropertyButton->setFlat( true ); + vboxl->addWidget( addPropertyButton ); + removePropertyButton = new TQPushButton( this ); + removePropertyButton->setText( i18n( "Delete" ) ); + removePropertyButton->setFlat( true ); + vboxl->addWidget( removePropertyButton ); + vboxl->addStretch(); + layout->addLayout( vboxl ); + + // Connect signals & slots + connect( addPropertyButton, TQ_SIGNAL( clicked() ), propertyListView, TQ_SLOT( createNew() ) ); + connect( removePropertyButton, TQ_SIGNAL( clicked() ), propertyListView, TQ_SLOT( remove + () ) ); + + //FIXME: We've got some sort of build issue... we get undefined references to CreatePropertyDialog without this dummy code here + UnitList list; + CreatePropertyDialog d( this, &list ); +} + + +PropertiesDialog::~PropertiesDialog() +{} + +void PropertiesDialog::reload( void ) +{ + propertyListView->reload(); +} + +#include "propertiesdialog.moc" diff --git a/src/dialogs/propertiesdialog.h b/src/dialogs/propertiesdialog.h new file mode 100644 index 0000000..d6ae406 --- /dev/null +++ b/src/dialogs/propertiesdialog.h @@ -0,0 +1,47 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef PROPERTIESDIALOG_H +#define PROPERTIESDIALOG_H + +#include <tqlayout.h> +#include <tqpushbutton.h> +#include <tqvbox.h> +#include <tdelistview.h> + +class RecipeDB; +class StdPropertyListView; + +/** +@author Unai Garro +*/ +class PropertiesDialog: public TQWidget +{ + TQ_OBJECT +public: + PropertiesDialog( TQWidget *parent, RecipeDB *db ); + ~PropertiesDialog(); + void reload( void ); + +private: + // Variables + RecipeDB *database; + + // Widgets + TQGridLayout* layout; + TQPushButton* addPropertyButton; + TQPushButton* removePropertyButton; + StdPropertyListView* propertyListView; + +}; + +#endif diff --git a/src/dialogs/recipeimportdialog.cpp b/src/dialogs/recipeimportdialog.cpp new file mode 100644 index 0000000..cbc3dcf --- /dev/null +++ b/src/dialogs/recipeimportdialog.cpp @@ -0,0 +1,185 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "recipeimportdialog.h" + +#include <tdelocale.h> +#include <kpushbutton.h> +#include <tdelistview.h> +#include <kdebug.h> + +#include <tqvbox.h> +#include <tqlayout.h> +#include <tqheader.h> +#include <tqvariant.h> +#include <tqdict.h> + +#include "datablocks/recipe.h" + +RecipeImportDialog::RecipeImportDialog( const RecipeList &list, TQWidget *parent ) + : KDialogBase( parent, "RecipeImportDialog", true, i18n( "Import Recipes" ), + KDialogBase::Ok | KDialogBase::Cancel, KDialogBase::Ok ), + list_copy( list ) +{ + setButtonBoxOrientation( Vertical ); + + TQVBox *page = makeVBoxMainWidget(); + + kListView = new TDEListView( page ); + kListView->addColumn( i18n( "Recipes" ) ); + kListView->setProperty( "selectionMode", "NoSelection" ); + kListView->setRootIsDecorated( true ); + kListView->setAllColumnsShowFocus( true ); + + languageChange(); + + setInitialSize( TQSize( 600, 480 ).expandedTo( minimumSizeHint() ) ); + + loadListView(); +} + +RecipeImportDialog::~RecipeImportDialog() +{ + delete recipe_items; +} + +void RecipeImportDialog::languageChange() +{ +} + +void RecipeImportDialog::loadListView() +{ + CustomCheckListItem * head_item = new CustomCheckListItem( kListView, TQString( i18n( "All (%1)" ) ).arg( list_copy.count() ), TQCheckListItem::CheckBox ); + head_item->setOpen( true ); + + //get all categories + TQStringList categoryList; + + RecipeList::const_iterator recipe_it; + for ( recipe_it = list_copy.begin(); recipe_it != list_copy.end(); ++recipe_it ) { + for ( ElementList::const_iterator cat_it = ( *recipe_it ).categoryList.begin(); cat_it != ( *recipe_it ).categoryList.end(); ++cat_it ) { + if ( categoryList.contains( ( *cat_it ).name ) < 1 ) + categoryList << ( *cat_it ).name; + } + } + + //create all category check list items + TQDict<CustomCheckListItem> all_categories; + + TQStringList::iterator it; + for ( it = categoryList.begin(); it != categoryList.end(); ++it ) { + CustomCheckListItem *category_item = new CustomCheckListItem( head_item, *it, TQCheckListItem::CheckBox ); + //category_item->setOpen(true); + + all_categories.insert( *it, category_item ); + } + + //add recipes to category check list items + recipe_items = new TQMap<CustomCheckListItem*, RecipeList::const_iterator>; //we won't be able to identify a recipe later if we just put a value in here. The iterator will be unique so we'll use it. This is safe since the list is constant (iterators won't become invlalid). + + CustomCheckListItem *item = 0; + CustomCheckListItem *category_item = 0; + + for ( recipe_it = list_copy.begin(); recipe_it != list_copy.end(); ++recipe_it ) { + if ( ( *recipe_it ).categoryList.count() == 0 ) { + if ( !category_item ) //don't create this until there are recipes to put in it + { + category_item = new CustomCheckListItem( head_item, i18n( "Uncategorized" ), TQCheckListItem::CheckBox ); + all_categories.insert( i18n( "Uncategorized" ), category_item ); + } + CustomCheckListItem *item = new CustomCheckListItem( category_item, ( *recipe_it ).title, TQCheckListItem::CheckBox ); + recipe_items->insert( item, recipe_it ); + } + else { + for ( ElementList::const_iterator cat_it = ( *recipe_it ).categoryList.begin(); cat_it != ( *recipe_it ).categoryList.end(); ++cat_it ) { + + CustomCheckListItem *category_item = all_categories[ ( *cat_it ).name ]; + + item = new CustomCheckListItem( category_item, item, ( *recipe_it ).title, TQCheckListItem::CheckBox ); + recipe_items->insert( item, recipe_it ); + } + } + } + + //append the number of recipes in each category to the check list item text + TQDictIterator<CustomCheckListItem> categories_it( all_categories ); + for ( ; categories_it.current(); ++categories_it ) { + int count = 0; + for ( TQCheckListItem * it = static_cast<TQCheckListItem*>( categories_it.current() ->firstChild() ); it; it = static_cast<TQCheckListItem*>( it->nextSibling() ) ) { + count++; + } + categories_it.current() ->setText( 0, categories_it.current() ->text( 0 ) + TQString( " (%1)" ).arg( count ) ); + } + + head_item->setOn( true ); //this will check all recipes +} + +RecipeList RecipeImportDialog::getSelectedRecipes() +{ + RecipeList selected_recipes; + + TQValueList<RecipeList::const_iterator> already_included_recipes; + + TQMap<CustomCheckListItem*, RecipeList::const_iterator>::const_iterator it; + for ( it = recipe_items->begin(); it != recipe_items->end(); ++it ) { + if ( static_cast<CustomCheckListItem*>( it.key() ) ->isOn() && + ( already_included_recipes.contains( it.data() ) == 0 ) ) //make sure it isn't already in the list + { + already_included_recipes.prepend( it.data() ); + selected_recipes.prepend( *it.data() ); + } + } + + return selected_recipes; +} + +CustomCheckListItem::CustomCheckListItem( TQListView *parent, const TQString & s, Type t ) + : TQCheckListItem( parent, s, t ), m_locked( false ) +{} + +CustomCheckListItem::CustomCheckListItem( CustomCheckListItem *parent, const TQString & s, Type t ) + : TQCheckListItem( parent, s, t ), m_locked( false ) +{} + +CustomCheckListItem::CustomCheckListItem( TQCheckListItem *parent, TQCheckListItem *after, const TQString & s, Type t ) + : TQCheckListItem( parent, after, s, t ), m_locked( false ) +{} + +void CustomCheckListItem::stateChange( bool on ) +{ + if ( !m_locked ) { + for ( TQCheckListItem * it = static_cast<TQCheckListItem*>( firstChild() ); it; it = static_cast<TQCheckListItem*>( it->nextSibling() ) ) { + it->setOn( on ); + } + } + + if ( !on ) { + TQListViewItem * parent = this->parent(); + if ( parent && ( parent->rtti() == 1 ) ) { + CustomCheckListItem * item = static_cast<CustomCheckListItem*>( parent ); + item->setLocked( true ); + item->setOn( on ); + item->setLocked( false ); + } + } + + TQString thisText = text(0); + TQListViewItemIterator it( listView() ); + while ( it.current() ) { + if ( it.current()->rtti() == 1 && it.current()->text(0) == thisText ) { + CustomCheckListItem * item = static_cast<CustomCheckListItem*>( it.current() ); + item->setOn( on ); + } + ++it; + } +} + diff --git a/src/dialogs/recipeimportdialog.h b/src/dialogs/recipeimportdialog.h new file mode 100644 index 0000000..5dfdf2c --- /dev/null +++ b/src/dialogs/recipeimportdialog.h @@ -0,0 +1,80 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef RECIPEIMPORTDIALOG_H +#define RECIPEIMPORTDIALOG_H + +#include <tqvaluelist.h> +#include <tqmap.h> +#include <tqlistview.h> + +#include <kdialogbase.h> + +#include "datablocks/recipelist.h" + +class TDEListView; + +class TQListViewItem; + +class Recipe; +class CustomCheckListItem; + +/** + * @author Jason Kivlighn + */ +class RecipeImportDialog : public KDialogBase +{ +public: + RecipeImportDialog( const RecipeList &all_recipes, TQWidget *parent = 0 ); + ~RecipeImportDialog(); + + RecipeList getSelectedRecipes(); + +protected slots: + virtual void languageChange(); + +private: + void loadListView(); + + TDEListView* kListView; + + TQMap<CustomCheckListItem*, RecipeList::const_iterator> *recipe_items; + const RecipeList list_copy; +}; + +/** A specialized TQCheckListItem that sets the state of its children to its + * current state. + * @author Jason Kivlighn + */ +class CustomCheckListItem : public TQCheckListItem +{ +public: + CustomCheckListItem( TQListView *parent, const TQString &, Type ); + CustomCheckListItem( CustomCheckListItem *parent, const TQString &, Type ); + CustomCheckListItem( TQCheckListItem *parent, TQCheckListItem *after, const TQString &, Type ); + +protected: + virtual void stateChange( bool ); + bool locked() const + { + return m_locked; + } + void setLocked( bool b ) + { + m_locked = b; + } + +private: + bool m_locked; +}; + +#endif //RECIPEIMPORTDIALOG_H diff --git a/src/dialogs/recipeinputdialog.cpp b/src/dialogs/recipeinputdialog.cpp new file mode 100644 index 0000000..31ee38e --- /dev/null +++ b/src/dialogs/recipeinputdialog.cpp @@ -0,0 +1,1641 @@ +/*************************************************************************** +* Copyright (C) 2003-2005 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* Copyright (C) 2006 Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "recipeinputdialog.h" + +#include <tqstring.h> +#include <tqlayout.h> +#include <tqhbox.h> +#include <tqvbox.h> +#include <tqimage.h> +#include <tqmessagebox.h> +#include <tqtooltip.h> +#include <tqdatetimeedit.h> +#include <tqdragobject.h> +#include <tqbuttongroup.h> +#include <tqradiobutton.h> +#include <tqwidgetstack.h> +#include <tqpainter.h> + +#include <tdeapplication.h> +#include <tdecompletionbox.h> +#include <tdespell.h> +#include <kurl.h> +#include <tdefiledialog.h> +#include <tdelocale.h> +#include <tdemessagebox.h> +#include <kdebug.h> +#include <kled.h> +#include <kdialogbase.h> + +#include "selectauthorsdialog.h" +#include "resizerecipedialog.h" +#include "ingredientparserdialog.h" +#include "editratingdialog.h" +#include "createunitdialog.h" +#include "datablocks/recipe.h" +#include "datablocks/categorytree.h" +#include "datablocks/unit.h" +#include "datablocks/weight.h" +#include "backends/recipedb.h" +#include "selectcategoriesdialog.h" +#include "widgets/fractioninput.h" +#include "widgets/kretextedit.h" +#include "widgets/inglistviewitem.h" +#include "../widgets/ratingdisplaywidget.h" +#include "widgets/kwidgetlistbox.h" +#include "widgets/ingredientinputwidget.h" +#include "image.h" //Initializes default photo + +#include "profiling.h" + +enum ColorStatus { GreenStatus, RedStatus, YellowStatus }; + +ClickableLed::ClickableLed( TQWidget *parent ) : KLed(parent) +{ +} + +void ClickableLed::mouseReleaseEvent( TQMouseEvent* ) +{ + emit clicked(); +} + +ImageDropLabel::ImageDropLabel( TQWidget *parent, TQPixmap &_sourcePhoto ) : TQLabel( parent ), + sourcePhoto( _sourcePhoto ) +{ + setAcceptDrops( TRUE ); +} + +void ImageDropLabel::dragEnterEvent( TQDragEnterEvent* event ) +{ + event->accept( TQImageDrag::canDecode( event ) ); +} + +void ImageDropLabel::dropEvent( TQDropEvent* event ) +{ + TQImage image; + + if ( TQImageDrag::decode( event, image ) ) { + if ( ( image.width() > width() || image.height() > height() ) || ( image.width() < width() && image.height() < height() ) ) { + TQPixmap pm_scaled; + pm_scaled.convertFromImage( image.smoothScale( width(), height(), TQImage::ScaleMin ) ); + setPixmap( pm_scaled ); + + sourcePhoto = pm_scaled; // to save scaled later on + } + else { + setPixmap( image ); + sourcePhoto = image; + } + + emit changed(); + } +} + + +RecipeInputDialog::RecipeInputDialog( TQWidget* parent, RecipeDB *db ) : TQVBox( parent ) +{ + + // Adjust internal parameters + loadedRecipe = new Recipe(); + loadedRecipe->recipeID = -1; // No loaded recipe initially + loadedRecipe->title = TQString::null; + loadedRecipe->instructions = TQString::null; + database = db; + + TDEIconLoader *il = new TDEIconLoader; + + // Tabs + tabWidget = new TQTabWidget( this, "tabWidget" ); + tabWidget->setSizePolicy( TQSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::MinimumExpanding ) ); + + + //------- Recipe Tab ----------------- + // Recipe Photo + + recipeTab = new TQGroupBox( tabWidget ); + recipeTab->setFrameStyle( TQFrame::NoFrame ); + recipeTab->setSizePolicy( TQSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::MinimumExpanding ) ); + + + // Design the Dialog + TQGridLayout* recipeLayout = new TQGridLayout( recipeTab, 1, 1, 0, 0 ); + + // Border + TQSpacerItem* spacer_left = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::Minimum ); + recipeLayout->addItem( spacer_left, 1, 0 ); + TQSpacerItem* spacer_right = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::Minimum ); + recipeLayout->addItem( spacer_right, 1, 8 ); + TQSpacerItem* spacer_top = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum , TQSizePolicy::Fixed ); + recipeLayout->addItem( spacer_top, 0, 1 ); + TQSpacerItem* spacer_bottom = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum , TQSizePolicy::MinimumExpanding ); + recipeLayout->addItem( spacer_bottom, 8, 1 ); + + + TQPixmap image1( defaultPhoto ); + + photoLabel = new ImageDropLabel( recipeTab, sourcePhoto ); + photoLabel->setPixmap( image1 ); + photoLabel->setFixedSize( TQSize( 221, 166 ) ); + photoLabel->setSizePolicy( TQSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ) ); + photoLabel->setAlignment( TQt::AlignHCenter | TQt::AlignVCenter ); + recipeLayout->addMultiCellWidget( photoLabel, 3, 7, 1, 1 ); + + TQVBox *photoButtonsBox = new TQVBox( recipeTab ); + + changePhotoButton = new TQPushButton( photoButtonsBox ); + changePhotoButton->setSizePolicy( TQSizePolicy( TQSizePolicy::Preferred, TQSizePolicy::Ignored ) ); + changePhotoButton->setText( "..." ); + TQToolTip::add + ( changePhotoButton, i18n( "Select photo" ) ); + + TQPushButton *clearPhotoButton = new TQPushButton( photoButtonsBox ); + clearPhotoButton->setPixmap( il->loadIcon( "clear_left", TDEIcon::NoGroup, 16 ) ); + TQToolTip::add + ( clearPhotoButton, i18n( "Clear photo" ) ); + + recipeLayout->addMultiCellWidget( photoButtonsBox, 3, 7, 2, 2 ); + + + //Title->photo spacer + TQSpacerItem* title_photo = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + recipeLayout->addItem( title_photo, 2, 3 ); + + + // Title + TQVBox *titleBox = new TQVBox( recipeTab ); + titleBox->setSpacing( 5 ); + titleLabel = new TQLabel( i18n( "Recipe Name" ), titleBox ); + titleEdit = new KLineEdit( titleBox ); + titleEdit->setMinimumSize( TQSize( 360, 30 ) ); + titleEdit->setMaximumSize( TQSize( 10000, 30 ) ); + titleEdit->setSizePolicy( TQSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::Fixed ) ); + recipeLayout->addMultiCellWidget( titleBox, 1, 1, 1, 7 ); + + + // Photo ->author spacer + TQSpacerItem* title_spacer = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::Minimum ); + recipeLayout->addItem( title_spacer, 2, 1 ); + + // Author(s) & Categories + TQVBox *authorBox = new TQVBox( recipeTab ); // contains label and authorInput (input widgets) + authorBox->setSpacing( 5 ); + recipeLayout->addWidget( authorBox, 3, 4 ); + authorLabel = new TQLabel( i18n( "Authors" ), authorBox ); + TQHBox *authorInput = new TQHBox( authorBox ); // Contains input + button + + + authorShow = new KLineEdit( authorInput ); + authorShow->setReadOnly( true ); + authorShow->setMinimumSize( TQSize( 100, 20 ) ); + authorShow->setMaximumSize( TQSize( 10000, 20 ) ); + authorShow->setSizePolicy( TQSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::Fixed ) ); + + + addAuthorButton = new TQPushButton( authorInput ); + addAuthorButton->setText( "+" ); + addAuthorButton->setFixedSize( TQSize( 20, 20 ) ); + addAuthorButton->setFlat( true ); + + + TQSpacerItem* author_category = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::Minimum ); + recipeLayout->addItem( author_category, 3, 5 ); + + TQVBox *categoryBox = new TQVBox( recipeTab ); // Contains the label and categoryInput (input widgets) + categoryBox->setSpacing( 5 ); + categoryLabel = new TQLabel( i18n( "Categories" ), categoryBox ); + TQHBox *categoryInput = new TQHBox( categoryBox ); // Contains the input widgets + + categoryShow = new KLineEdit( categoryInput ); + categoryShow->setReadOnly( true ); + categoryShow->setMinimumSize( TQSize( 100, 20 ) ); + categoryShow->setMaximumSize( TQSize( 10000, 20 ) ); + categoryShow->setSizePolicy( TQSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::Fixed ) ); + recipeLayout->addWidget( categoryBox, 4, 4 ); + + addCategoryButton = new TQPushButton( categoryInput ); + addCategoryButton->setText( "+" ); + addCategoryButton->setFixedSize( TQSize( 20, 20 ) ); + addCategoryButton->setFlat( true ); + + //Category ->Servings spacer + TQSpacerItem* category_yield = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + recipeLayout->addItem( category_yield, 5, 4 ); + + TQHBox *serv_prep_box = new TQHBox( recipeTab ); + serv_prep_box->setSpacing( 5 ); + + // Backup options + TQGroupBox *yieldGBox = new TQGroupBox( serv_prep_box, "yieldGBox" ); + yieldGBox->setTitle( i18n( "Yield" ) ); + yieldGBox->setColumns( 2 ); + + yieldLabel = new TQLabel( i18n( "Amount" ), yieldGBox ); + /*TQLabel *yieldTypeLabel = */new TQLabel( i18n( "Type" ), yieldGBox ); + yieldNumInput = new FractionInput( yieldGBox ); + yieldNumInput->setAllowRange(true); + yieldTypeEdit = new KLineEdit( yieldGBox ); + + TQVBox *prepTimeBox = new TQVBox( serv_prep_box ); + prepTimeBox->setSizePolicy( TQSizePolicy( TQSizePolicy::Minimum, TQSizePolicy::Fixed ) ); + prepTimeBox->setSpacing( 5 ); + + ( void ) new TQLabel( i18n( "Preparation Time" ), prepTimeBox ); + prepTimeEdit = new TQTimeEdit( prepTimeBox ); + prepTimeEdit->setMinValue( TQTime( 0, 0 ) ); + prepTimeEdit->setDisplay( TQTimeEdit::Hours | TQTimeEdit::Minutes ); + + recipeLayout->addWidget( serv_prep_box, 6, 4 ); + + //------- END OF Recipe Tab --------------- + + //------- Ingredients Tab ----------------- + + ingredientGBox = new TQGroupBox( recipeTab ); + ingredientGBox->setFrameStyle( TQFrame::NoFrame ); + ingredientGBox->setSizePolicy( TQSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::MinimumExpanding ) ); + TQGridLayout* ingredientsLayout = new TQGridLayout( ingredientGBox ); + + // Border + TQSpacerItem* spacerBoxLeft = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::Minimum ); + ingredientsLayout->addItem( spacerBoxLeft, 1, 0 ); + TQSpacerItem* spacerBoxTop = new TQSpacerItem( 10, 20, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + ingredientsLayout->addItem( spacerBoxTop, 0, 1 ); + + //Input Widgets + ingInput = new IngredientInputWidget( database, ingredientGBox ); + ingredientsLayout->addMultiCellWidget( ingInput, 1, 1, 1, 5 ); + + // Spacers to list and buttons + TQSpacerItem* spacerToList = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + ingredientsLayout->addItem( spacerToList, 2, 1 ); + TQSpacerItem* spacerToButtons = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::Minimum ); + ingredientsLayout->addItem( spacerToButtons, 3, 4 ); + + // Add, Up,down,... buttons + + addButton = new KPushButton( ingredientGBox ); + addButton->setFixedSize( TQSize( 31, 31 ) ); + addButton->setFlat( true ); + TQPixmap pm = il->loadIcon( "new", TDEIcon::NoGroup, 16 ); + addButton->setPixmap( pm ); + addButton->setSizePolicy( TQSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ) ); + ingredientsLayout->addWidget( addButton, 3, 5 ); + + // Spacer to the rest of buttons + TQSpacerItem* spacerToOtherButtons = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + ingredientsLayout->addItem( spacerToOtherButtons, 4, 5 ); + + upButton = new KPushButton( ingredientGBox ); + upButton->setFixedSize( TQSize( 31, 31 ) ); + upButton->setFlat( true ); + pm = il->loadIcon( "go-up", TDEIcon::NoGroup, 16 ); + upButton->setPixmap( pm ); + upButton->setSizePolicy( TQSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ) ); + ingredientsLayout->addWidget( upButton, 5, 5 ); + + downButton = new KPushButton( ingredientGBox ); + downButton->setFixedSize( TQSize( 31, 31 ) ); + downButton->setFlat( true ); + pm = il->loadIcon( "go-down", TDEIcon::NoGroup, 16 ); + downButton->setPixmap( pm ); + downButton->setSizePolicy( TQSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ) ); + ingredientsLayout->addWidget( downButton, 6, 5 ); + + removeButton = new KPushButton( ingredientGBox ); + removeButton->setFixedSize( TQSize( 31, 31 ) ); + removeButton->setFlat( true ); + pm = il->loadIcon( "remove", TDEIcon::NoGroup, 16 ); + removeButton->setPixmap( pm ); + removeButton->setSizePolicy( TQSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ) ); + ingredientsLayout->addWidget( removeButton, 7, 5 ); + + ingParserButton = new KPushButton( ingredientGBox ); + ingParserButton->setFixedSize( TQSize( 31, 31 ) ); + ingParserButton->setFlat( true ); + pm = il->loadIcon( "edit-paste", TDEIcon::NoGroup, 16 ); + ingParserButton->setPixmap( pm ); + ingParserButton->setSizePolicy( TQSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ) ); + ingredientsLayout->addWidget( ingParserButton, 8, 5 ); + + TQToolTip::add + ( addButton, i18n( "Add ingredient" ) ); + TQToolTip::add + ( upButton, i18n( "Move ingredient up" ) ); + TQToolTip::add + ( downButton, i18n( "Move ingredient down" ) ); + TQToolTip::add + ( removeButton, i18n( "Remove ingredient" ) ); + TQToolTip::add + ( ingParserButton, i18n( "Paste Ingredients" ) ); + + // Ingredient List + ingredientList = new TDEListView( ingredientGBox, "ingredientList" ); + ingredientList->addColumn( i18n( "Ingredient" ) ); + ingredientList->addColumn( i18n( "Amount" ) ); + ingredientList->setColumnAlignment( 1, TQt::AlignHCenter ); + ingredientList->addColumn( i18n( "Units" ) ); + ingredientList->addColumn( i18n( "Preparation Method" ) ); + ingredientList->setSorting( -1 ); // Do not sort + ingredientList->setSizePolicy( TQSizePolicy( TQSizePolicy::Expanding, TQSizePolicy::MinimumExpanding ) ); + ingredientList->setItemsRenameable( true ); + ingredientList->setRenameable( 0, false ); //name + ingredientList->setRenameable( 1, true ); //amount + ingredientList->setRenameable( 2, true ); //units + ingredientList->setRenameable( 3, true ); //prep method + ingredientList->setDefaultRenameAction( TQListView::Reject ); + ingredientsLayout->addMultiCellWidget( ingredientList, 3, 9, 1, 3 ); + + TQHBoxLayout *propertyStatusLayout = new TQHBoxLayout( NULL, 0, 5 ); + TQLabel *propertyLabel = new TQLabel( i18n("Property Status:"), ingredientGBox ); + propertyStatusLabel = new TQLabel( ingredientGBox ); + propertyStatusLed = new ClickableLed( ingredientGBox ); + propertyStatusLed->setFixedSize( TQSize(16,16) ); + propertyStatusButton = new TQPushButton( i18n("Details..."), ingredientGBox ); + //TQPushButton *propertyUpdateButton = new TQPushButton( i18n("Update"), ingredientGBox ); + propertyStatusLayout->addWidget( propertyLabel ); + propertyStatusLayout->addWidget( propertyStatusLabel ); + propertyStatusLayout->addWidget( propertyStatusLed ); + propertyStatusLayout->addWidget( propertyStatusButton ); + //propertyStatusLayout->addWidget( propertyUpdateButton ); + TQSpacerItem* propertySpacerRight = new TQSpacerItem( 10, 10, TQSizePolicy::MinimumExpanding, TQSizePolicy::Minimum ); + propertyStatusLayout->addItem( propertySpacerRight ); + + KGuiItem updateGuiItem; + updateGuiItem.setText( i18n("Update") ); + updateGuiItem.setIconSet( il->loadIconSet( "reload", TDEIcon::NoGroup ) ); + propertyStatusDialog = new KDialogBase( KDialogBase::Swallow, i18n("Property details"), + KDialogBase::Close | KDialogBase::User1 | KDialogBase::Help, + KDialogBase::Close, this, "propertyStatusDialog", false, false, + updateGuiItem + ); + propertyStatusDialog->setHelp("property-status"); + statusTextView = new TQTextEdit(0); + statusTextView->setTextFormat( TQt::RichText ); + statusTextView->setReadOnly(true); + propertyStatusDialog->setMainWidget( statusTextView ); + propertyStatusDialog->resize( 400, 300 ); + + ingredientsLayout->addMultiCellLayout( propertyStatusLayout, 10, 10, 1, 4 ); + + // ------- Recipe Instructions Tab ----------- + + instructionsTab = new TQGroupBox( recipeTab ); + instructionsTab->setFrameStyle( TQFrame::NoFrame ); + instructionsTab->setSizePolicy( TQSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::MinimumExpanding ) ); + + TQVBoxLayout *instructionsLayout = new TQVBoxLayout( instructionsTab ); + + instructionsEdit = new KreTextEdit( instructionsTab ); + instructionsEdit->setSizePolicy( TQSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::MinimumExpanding ) ); + instructionsEdit->setTabChangesFocus ( true ); + instructionsLayout->addWidget( instructionsEdit ); + + spellCheckButton = new TQToolButton( instructionsTab ); + spellCheckButton->setIconSet( il->loadIconSet( "tools-check-spelling", TDEIcon::Small ) ); + TQToolTip::add + ( spellCheckButton, i18n( "Check spelling" ) ); + instructionsLayout->addWidget( spellCheckButton ); + + // ------- END OF Recipe Instructions Tab ----------- + + + // ------- Recipe Ratings Tab ----------- + + TQVBox *ratingsTab = new TQVBox(recipeTab); + ratingListDisplayWidget = new KWidgetListbox(ratingsTab); + TQPushButton *addRatingButton = new TQPushButton(i18n("Add Rating..."),ratingsTab); + + connect( addRatingButton, TQ_SIGNAL(clicked()), this, TQ_SLOT(slotAddRating()) ); + + // ------- END OF Recipe Ratings Tab ----------- + + + tabWidget->insertTab( recipeTab, i18n( "Recipe" ) ); + tabWidget->insertTab( ingredientGBox, i18n( "Ingredients" ) ); + tabWidget->insertTab( instructionsTab, i18n( "Instructions" ) ); + tabWidget->insertTab( ratingsTab, i18n( "Ratings" ) ); + + + // Functions Box + TQHBox* functionsLayout = new TQHBox( this ); + + functionsBox = new TQGroupBox( 1, TQt::Vertical, functionsLayout ); + functionsBox->setFrameStyle( TQFrame::NoFrame ); + + saveButton = new TQToolButton( functionsBox ); + saveButton->setIconSet( il->loadIconSet( "document-save", TDEIcon::Small ) ); + saveButton->setEnabled( false ); + showButton = new TQToolButton( functionsBox ); + showButton->setIconSet( il->loadIconSet( "viewmag", TDEIcon::Small ) ); + closeButton = new TQToolButton( functionsBox ); + closeButton->setIconSet( il->loadIconSet( "window-close", TDEIcon::Small ) ); + resizeButton = new TQToolButton( functionsBox ); + resizeButton->setIconSet( il->loadIconSet( "2uparrow", TDEIcon::Small ) ); //TODO: give me an icon :) + + saveButton->setTextLabel( i18n( "Save recipe" ), true ); + saveButton->setUsesTextLabel( true ); + showButton->setTextLabel( i18n( "Show recipe" ), true ); + showButton->setUsesTextLabel( true ); + closeButton->setTextLabel( i18n( "Close" ), true ); + closeButton->setUsesTextLabel( true ); + resizeButton->setTextLabel( i18n( "Resize recipe" ), true ); + resizeButton->setUsesTextLabel( true ); + + functionsLayout->layout() ->addItem( new TQSpacerItem( 10, 10, TQSizePolicy::MinimumExpanding, TQSizePolicy::Fixed ) ); + + // Dialog design + tabWidget->resize( size().expandedTo( minimumSizeHint() ) ); + clearWState( WState_Polished ); + + // Initialize internal data + unsavedChanges = false; // Indicates if there's something not saved yet. + enableChangedSignal(); // Enables the signal "changed()" + + // Connect signals & Slots + connect( changePhotoButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( changePhoto() ) ); + connect( clearPhotoButton, TQ_SIGNAL( clicked() ), TQ_SLOT( clearPhoto() ) ); + connect( upButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( moveIngredientUp() ) ); + connect( downButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( moveIngredientDown() ) ); + connect( removeButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( removeIngredient() ) ); + connect( addButton, TQ_SIGNAL( clicked() ), ingInput, TQ_SLOT( addIngredient() ) ); + connect( ingParserButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( slotIngredientParser() ) ); + connect( photoLabel, TQ_SIGNAL( changed() ), this, TQ_SIGNAL( changed() ) ); + connect( this, TQ_SIGNAL( changed() ), this, TQ_SLOT( recipeChanged() ) ); + connect( yieldNumInput, TQ_SIGNAL( textChanged( const TQString & ) ), this, TQ_SLOT( recipeChanged() ) ); + connect( yieldTypeEdit, TQ_SIGNAL( textChanged( const TQString & ) ), this, TQ_SLOT( recipeChanged() ) ); + connect( prepTimeEdit, TQ_SIGNAL( valueChanged( const TQTime & ) ), TQ_SLOT( recipeChanged() ) ); + connect( titleEdit, TQ_SIGNAL( textChanged( const TQString& ) ), this, TQ_SLOT( recipeChanged( const TQString& ) ) ); + connect( instructionsEdit, TQ_SIGNAL( textChanged() ), this, TQ_SLOT( recipeChanged() ) ); + connect( addCategoryButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( addCategory() ) ); + connect( addAuthorButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( addAuthor() ) ); + connect( titleEdit, TQ_SIGNAL( textChanged( const TQString& ) ), this, TQ_SLOT( prepTitleChanged( const TQString& ) ) ); + connect( ingredientList, TQ_SIGNAL( itemRenamed( TQListViewItem*, const TQString &, int ) ), TQ_SLOT( syncListView( TQListViewItem*, const TQString &, int ) ) ); + + connect ( ingInput, TQ_SIGNAL( ingredientEntered(const Ingredient&) ), this, TQ_SLOT( addIngredient(const Ingredient&) ) ); + connect ( ingInput, TQ_SIGNAL( headerEntered(const Element&) ), this, TQ_SLOT( addIngredientHeader(const Element&) ) ); + + connect( propertyStatusLed, TQ_SIGNAL(clicked()), TQ_SLOT(updatePropertyStatus()) ); + connect( propertyStatusDialog, TQ_SIGNAL(user1Clicked()), TQ_SLOT(updatePropertyStatus()) ); + connect( propertyStatusButton, TQ_SIGNAL(clicked()), propertyStatusDialog, TQ_SLOT(show()) ); + + // Function buttons + connect ( saveButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( save() ) ); + connect ( closeButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( closeOptions() ) ); + connect ( showButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( showRecipe() ) ); + connect ( resizeButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( resizeRecipe() ) ); + connect ( spellCheckButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( spellCheck() ) ); + connect ( this, TQ_SIGNAL( enableSaveOption( bool ) ), this, TQ_SLOT( enableSaveButton( bool ) ) ); + + connect ( database, TQ_SIGNAL( recipeRemoved(int) ), this, TQ_SLOT( recipeRemoved(int) ) ); + + delete il; + + //FIXME: We've got some sort of build issue... we get undefined references to CreateUnitDialog without this dummy code here + CreateUnitDialog d( this, "" ); +} + + +RecipeInputDialog::~RecipeInputDialog() +{ + delete loadedRecipe; +} + +void RecipeInputDialog::recipeRemoved( int id ) +{ + if ( loadedRecipe->recipeID == id ) { + loadedRecipe->recipeID = -1; + recipeChanged(); + } +} + +void RecipeInputDialog::prepTitleChanged( const TQString &title ) +{ + //we don't want the menu to grow due to a long title + //### KStringHandler::rsqueeze does this but I can't remember when it was added (compatibility issue...) + TQString short_title = title.left( 20 ); + if ( title.length() > 20 ) + short_title.append( "..." ); + + emit titleChanged( short_title ); +} + +int RecipeInputDialog::loadedRecipeID() const +{ + return loadedRecipe->recipeID; +} + +void RecipeInputDialog::loadRecipe( int recipeID ) +{ + emit enableSaveOption( false ); + unsavedChanges = false; + + //Disable changed() signals + enableChangedSignal( false ); + + //Empty current recipe + loadedRecipe->empty(); + + //Set back to the first page + tabWidget->setCurrentPage( 0 ); + + // Load specified Recipe ID + database->loadRecipe( loadedRecipe, RecipeDB::All ^ RecipeDB::Meta ^ RecipeDB::Properties, recipeID ); + + reload(); + + propertyStatusDialog->hide(); + updatePropertyStatus(); + + //Enable changed() signals + enableChangedSignal(); + +} + +void RecipeInputDialog::reload( void ) +{ + yieldNumInput->setValue( 1, 0 ); + yieldTypeEdit->setText(""); + ingredientList->clear(); + ratingListDisplayWidget->clear(); + ingInput->clear(); + + //Load Values in Interface + titleEdit->setText( loadedRecipe->title ); + instructionsEdit->setText( loadedRecipe->instructions ); + yieldNumInput->setValue( loadedRecipe->yield.amount, loadedRecipe->yield.amount_offset ); + yieldTypeEdit->setText( loadedRecipe->yield.type ); + prepTimeEdit->setTime( loadedRecipe->prepTime ); + + //show ingredient list + IngredientList list_copy = loadedRecipe->ingList; + for ( IngredientList group_list = list_copy.firstGroup(); group_list.count() != 0; group_list = list_copy.nextGroup() ) { + TQListViewItem * lastElement = ingredientList->lastItem(); + TQListViewItem *ing_header = 0; + + TQString group = group_list[ 0 ].group; + if ( !group.isEmpty() ) { + if ( lastElement && lastElement->parent() ) + lastElement = lastElement->parent(); + + ing_header = new IngGrpListViewItem( ingredientList, lastElement, group_list[ 0 ].group, group_list[ 0 ].groupID ); + ing_header->setOpen( true ); + lastElement = ing_header; + } + + for ( IngredientList::const_iterator ing_it = group_list.begin(); ing_it != group_list.end(); ++ing_it ) { + //Insert ingredient after last one + if ( ing_header ) { + lastElement = new IngListViewItem ( ing_header, lastElement, *ing_it ); + } + else { + if ( lastElement && lastElement->parent() ) + lastElement = lastElement->parent(); + lastElement = new IngListViewItem ( ingredientList, lastElement, *ing_it ); + } + + for ( TQValueList<IngredientData>::const_iterator sub_it = (*ing_it).substitutes.begin(); sub_it != (*ing_it).substitutes.end(); ++sub_it ) { + new IngSubListViewItem ( lastElement, *sub_it ); + lastElement->setOpen(true); + } + + //update completion + instructionsEdit->addCompletionItem( ( *ing_it ).name ); + } + } + // + //show photo + if ( !loadedRecipe->photo.isNull() ) { + + // //get the photo + sourcePhoto = loadedRecipe->photo; + + if ( ( sourcePhoto.width() > photoLabel->width() || sourcePhoto.height() > photoLabel->height() ) || ( sourcePhoto.width() < photoLabel->width() && sourcePhoto.height() < photoLabel->height() ) ) { + TQImage pm = sourcePhoto.convertToImage(); + TQPixmap pm_scaled; + pm_scaled.convertFromImage( pm.smoothScale( photoLabel->width(), photoLabel->height(), TQImage::ScaleMin ) ); + photoLabel->setPixmap( pm_scaled ); + + sourcePhoto = pm_scaled; // to save scaled later on + } + else { + photoLabel->setPixmap( sourcePhoto ); + } + } + else { + TQPixmap photo = TQPixmap( defaultPhoto ); + photoLabel->setPixmap( photo ); + sourcePhoto = TQPixmap(); + } + + + // Show categories + showCategories(); + + // Show authors + showAuthors(); + + // Show ratings + for ( RatingList::iterator rating_it = loadedRecipe->ratingList.begin(); rating_it != loadedRecipe->ratingList.end(); ++rating_it ) { + RatingDisplayWidget *item = new RatingDisplayWidget; + item->rating_it = rating_it; + addRating(*rating_it,item); + ratingListDisplayWidget->insertItem(item); + } + ratingListDisplayWidget->ensureCellVisible(0,0); + + // Update yield type auto completion + TDECompletion *completion = yieldTypeEdit->completionObject(); + completion->clear(); + ElementList yieldList; + database->loadYieldTypes( &yieldList ); + for ( ElementList::const_iterator it = yieldList.begin(); it != yieldList.end(); ++it ) { + completion->addItem( (*it).name ); + } +} + +void RecipeInputDialog::changePhoto( void ) +{ + // standard filedialog + KURL filename = KFileDialog::getOpenURL( TQString::null, TQString( "*.png *.jpg *.jpeg *.xpm *.gif|%1 (*.png *.jpg *.jpeg *.xpm *.gif)" ).arg( i18n( "Images" ) ), this ); + TQPixmap pixmap ( filename.path() ); + if ( !( pixmap.isNull() ) ) { + // If photo is bigger than the label, or smaller in width, than photoLabel, scale it + sourcePhoto = pixmap; + if ( ( sourcePhoto.width() > photoLabel->width() || sourcePhoto.height() > photoLabel->height() ) || ( sourcePhoto.width() < photoLabel->width() && sourcePhoto.height() < photoLabel->height() ) ) { + TQImage pm = sourcePhoto.convertToImage(); + TQPixmap pm_scaled; + pm_scaled.convertFromImage( pm.smoothScale( photoLabel->width(), photoLabel->height(), TQImage::ScaleMin ) ); + photoLabel->setPixmap( pm_scaled ); + + sourcePhoto = pm_scaled; // to save scaled later on + photoLabel->setPixmap( pm_scaled ); + } + else { + photoLabel->setPixmap( sourcePhoto ); + } + emit changed(); + } +} + +void RecipeInputDialog::clearPhoto( void ) +{ + sourcePhoto = TQPixmap(); + photoLabel->setPixmap( TQPixmap( defaultPhoto ) ); + + emit changed(); +} + +void RecipeInputDialog::moveIngredientUp( void ) +{ + TQListViewItem * it = ingredientList->selectedItem(); + if ( !it || it->rtti() == INGSUBLISTVIEWITEM_RTTI ) + return ; + + TQListViewItem *iabove = it->itemAbove(); + while ( iabove && iabove->rtti() == INGSUBLISTVIEWITEM_RTTI ) + iabove = iabove->itemAbove(); + + if ( iabove ) { + if ( it->rtti() == INGGRPLISTVIEWITEM_RTTI ) { + if ( iabove->parent() ) + iabove = iabove->parent(); + + int it_index = ingItemIndex( ingredientList, it ); + int iabove_index = ingItemIndex( ingredientList, iabove ); + + iabove->moveItem( it ); //Move the Item + + loadedRecipe->ingList.move( iabove_index, ( iabove->rtti() == INGGRPLISTVIEWITEM_RTTI ) ? iabove->childCount() : 1, it_index + it->childCount() - 1 ); + } + else { + int it_index = ingItemIndex( ingredientList, it ); + int iabove_index = ingItemIndex( ingredientList, iabove ); + IngredientList::iterator ing = loadedRecipe->ingList.at( it_index ); + + if ( iabove->parent() != it->parent() ) { + if ( iabove->rtti() == INGGRPLISTVIEWITEM_RTTI && it->parent() ) { //move the item out of the group + it->parent() ->takeItem( it ); + ingredientList->insertItem( it ); + it->moveItem( ( iabove->itemAbove() ->parent() ) ? iabove->itemAbove() ->parent() : iabove->itemAbove() ); //Move the Item + } + else { //move the item into the group + ingredientList->takeItem( it ); + iabove->parent() ->insertItem( it ); + it->moveItem( iabove ); //Move the Item + } + + ingredientList->setCurrentItem( it ); //Keep selected + } + else { + iabove->moveItem( it ); //Move the Item + loadedRecipe->ingList.move( it_index, iabove_index ); + } + + if ( it->parent() ) + ( *ing ).groupID = ( ( IngGrpListViewItem* ) it->parent() ) ->id(); + else + ( *ing ).groupID = -1; + } + + emit changed(); + } +} + +void RecipeInputDialog::moveIngredientDown( void ) +{ + TQListViewItem * it = ingredientList->selectedItem(); + if ( !it || it->rtti() == INGSUBLISTVIEWITEM_RTTI ) + return ; + + TQListViewItem *ibelow = it->itemBelow(); + while ( ibelow && ibelow->rtti() == INGSUBLISTVIEWITEM_RTTI ) + ibelow = ibelow->itemBelow(); + + if ( ibelow ) { + if ( it->rtti() == INGGRPLISTVIEWITEM_RTTI ) { + TQListViewItem * next_sibling = it->nextSibling(); + + if ( next_sibling ) { + int it_index = ingItemIndex( ingredientList, it ); + int ibelow_index = ingItemIndex( ingredientList, next_sibling ); + + it->moveItem( next_sibling ); //Move the Item + + int skip = 0; + if ( next_sibling->childCount() > 0 ) + skip = next_sibling->childCount() - 1; + + loadedRecipe->ingList.move( it_index, it->childCount(), ibelow_index + skip ); + } + } + else { + int it_index = ingItemIndex( ingredientList, it ); + int ibelow_index = ingItemIndex( ingredientList, ibelow ); + IngredientList::iterator ing = loadedRecipe->ingList.at( it_index ); + + if ( ibelow->rtti() == INGGRPLISTVIEWITEM_RTTI || ( ibelow->parent() != it->parent() ) ) { + if ( ibelow->rtti() == INGGRPLISTVIEWITEM_RTTI && !it->parent() ) { //move the item into the group + if ( !it->parent() ) + ingredientList->takeItem( it ); + else + it->parent() ->takeItem( it ); + + ibelow->insertItem( it ); + } + else { //move the item out of the group + TQListViewItem *parent = it->parent(); //store this because we can't get it after we do it->takeItem() + parent->takeItem( it ); + ingredientList->insertItem( it ); + it->moveItem( parent ); //Move the Item + } + + ingredientList->setCurrentItem( it ); //Keep selected + } + else { + it->moveItem( ibelow ); //Move the Item + loadedRecipe->ingList.move( it_index, ibelow_index ); + } + + if ( it->parent() ) + ( *ing ).groupID = ( ( IngGrpListViewItem* ) it->parent() ) ->id(); + else + ( *ing ).groupID = -1; + } + + emit changed(); + } + else if ( it->parent() ) { + it->parent() ->takeItem( it ); + ingredientList->insertItem( it ); + it->moveItem( ( ingredientList->lastItem() ->parent() ) ? ingredientList->lastItem() ->parent() : ingredientList->lastItem() ); //Move the Item + ingredientList->setCurrentItem( it ); //Keep selected + + int it_index = ingItemIndex( ingredientList, it ); + IngredientList::iterator ing = loadedRecipe->ingList.at( it_index ); + ( *ing ).groupID = -1; + + emit changed(); + } +} + +void RecipeInputDialog::removeIngredient( void ) +{ + TQListViewItem * it = ingredientList->selectedItem(); + if ( it && (it->rtti() == INGLISTVIEWITEM_RTTI || it->rtti() == INGSUBLISTVIEWITEM_RTTI) ) { + TQListViewItem *iselect = it->itemBelow(); + while ( iselect && iselect->rtti() == INGSUBLISTVIEWITEM_RTTI ) + iselect = iselect->itemBelow(); + + if ( !iselect ) { + iselect = it->itemAbove(); + while ( iselect && iselect->rtti() == INGSUBLISTVIEWITEM_RTTI ) + iselect = iselect->itemAbove(); + } + + IngListViewItem *ing_item = (IngListViewItem*)it; //we can cast IngSubListViewItem to this too, it's a subclass + + IngredientData &ing = loadedRecipe->ingList.findSubstitute( ing_item->ingredient() ); + + //Remove it from the instruction's completion + instructionsEdit->removeCompletionItem( ing.name ); + + loadedRecipe->ingList.removeSubstitute( ing ); + + int ingID = ing_item->ingredient().ingredientID; + TQMap<int,TQString>::iterator map_it; + if ( (map_it = propertyStatusMapRed.find(ingID)) != propertyStatusMapRed.end() ) + propertyStatusMapRed.remove( map_it ); + else if ( (map_it = propertyStatusMapYellow.find(ingID)) != propertyStatusMapYellow.end() ) + propertyStatusMapYellow.remove( map_it ); + showStatusIndicator(); + + //Now remove the ingredient + it->setSelected( false ); + delete it; + if ( iselect ) + ingredientList->setSelected( iselect, true ); // be careful iselect->setSelected doesn't work this way. + + emit changed(); + } + else if ( it && it->rtti() == INGGRPLISTVIEWITEM_RTTI ) { + IngGrpListViewItem * header = ( IngGrpListViewItem* ) it; + + for ( IngListViewItem * sub_item = (IngListViewItem*)header->firstChild(); sub_item; sub_item = (IngListViewItem*)sub_item->nextSibling() ) { + IngredientData &ing = loadedRecipe->ingList.findSubstitute( sub_item->ingredient() ); + + //Remove it from the instruction's completion + instructionsEdit->removeCompletionItem( ing.name ); + + loadedRecipe->ingList.removeSubstitute( ing ); + + int ingID = sub_item->ingredient().ingredientID; + TQMap<int,TQString>::iterator map_it; + if ( (map_it = propertyStatusMapRed.find(ingID)) != propertyStatusMapRed.end() ) + propertyStatusMapRed.remove( map_it ); + else if ( (map_it = propertyStatusMapYellow.find(ingID)) != propertyStatusMapYellow.end() ) + propertyStatusMapYellow.remove( map_it ); + showStatusIndicator(); + } + + delete header; + + emit changed(); + } + +} + +int RecipeInputDialog::createNewYieldIfNecessary( const TQString &yield ) +{ + if ( yield.stripWhiteSpace().isEmpty() ) //no yield + return -1; + else + { + int id = database->findExistingYieldTypeByName( yield ); + if ( id == -1 ) //creating new + { + database->createNewYieldType( yield ); + id = database->lastInsertID(); + } + + return id; + } +} + +void RecipeInputDialog::syncListView( TQListViewItem* it, const TQString &new_text, int col ) +{ + if ( it->rtti() != INGLISTVIEWITEM_RTTI ) + return ; + + IngListViewItem *ing_item = ( IngListViewItem* ) it; + + IngredientData &new_ing = loadedRecipe->ingList.findSubstitute( ing_item->ingredient() ); + + switch ( col ) { + case 1: //amount + { + bool ok; + + Ingredient new_ing_amount; + new_ing_amount.setAmount(new_text,&ok); + + if ( ok ) + { + if ( new_ing.amount != new_ing_amount.amount || + new_ing.amount_offset != new_ing_amount.amount_offset ) { + new_ing.amount = new_ing_amount.amount; + new_ing.amount_offset = new_ing_amount.amount_offset; + if ( !new_text.isEmpty() ) + ing_item->setAmount( new_ing_amount.amount, new_ing_amount.amount_offset ); + + new_ing.amount = new_ing_amount.amount; + new_ing.amount_offset = new_ing_amount.amount_offset; + emit changed(); + } + } + else + { + if ( !new_text.isEmpty() ) + ing_item->setAmount( new_ing.amount, new_ing.amount_offset ); + } + + break; + } + case 2: //unit + { + Unit old_unit = new_ing.units; + + if ( new_text.length() > uint(database->maxUnitNameLength()) ) + { + KMessageBox::error( this, TQString( i18n( "Unit name cannot be longer than %1 characters." ) ).arg( database->maxUnitNameLength() ) ); + ing_item->setUnit( old_unit ); + break; + } + + TQString approp_unit = new_ing.amount > 1 ? new_ing.units.plural : new_ing.units.name; + if ( approp_unit != new_text.stripWhiteSpace() ) + { + Unit new_unit; + int new_id = IngredientInputWidget::createNewUnitIfNecessary( new_text.stripWhiteSpace(), new_ing.amount > 1, ing_item->ingredient().ingredientID, new_unit, database ); + + if ( new_id != -1 ) { + new_ing.units = new_unit; + new_ing.units.id = new_id; + + ing_item->setUnit( new_ing.units ); + + updatePropertyStatus(); + emit changed(); + } + else { + ing_item->setUnit( old_unit ); + } + } + break; + } + case 3: //prep method + { + TQString old_text = new_ing.prepMethodList.join(","); + + TQStringList prepMethodList = TQStringList::split(",",new_text.stripWhiteSpace()); + + for ( TQStringList::const_iterator it = prepMethodList.begin(); it != prepMethodList.end(); ++it ) { + if ( (*it).stripWhiteSpace().length() > uint(database->maxPrepMethodNameLength()) ) + { + KMessageBox::error( this, TQString( i18n( "Preparation method cannot be longer than %1 characters." ) ).arg( database->maxPrepMethodNameLength() ) ); + ing_item->setPrepMethod( old_text ); + break; + } + } + + if ( old_text != new_text.stripWhiteSpace() ) + { + new_ing.prepMethodList = ElementList::split(",",new_text.stripWhiteSpace()); + TQValueList<int> new_ids = IngredientInputWidget::createNewPrepIfNecessary( new_ing.prepMethodList, database ); + + TQValueList<int>::const_iterator id_it = new_ids.begin(); + for ( ElementList::iterator it = new_ing.prepMethodList.begin(); it != new_ing.prepMethodList.end(); ++it, ++id_it ) { + (*it).id = *id_it; + } + + updatePropertyStatus(); + emit changed(); + } + break; + } + } +} + +void RecipeInputDialog::recipeChanged( void ) +{ + if ( changedSignalEnabled ) { + // Enable Save Button + emit enableSaveOption( true ); + emit createButton( this, titleEdit->text() ); + unsavedChanges = true; + + } + +} + +void RecipeInputDialog::recipeChanged( const TQString & /*t*/ ) +{ + recipeChanged(); // jumps to the real slot function +} + +void RecipeInputDialog::enableChangedSignal( bool en ) +{ + changedSignalEnabled = en; +} + +bool RecipeInputDialog::save ( void ) +{ + //check bounds first + if ( titleEdit->text().length() > uint(database->maxRecipeTitleLength()) ) { + KMessageBox::error( this, TQString( i18n( "Recipe title cannot be longer than %1 characters." ) ).arg( database->maxRecipeTitleLength() ), i18n( "Unable to save recipe" ) ); + return false; + } + + emit enableSaveOption( false ); + saveRecipe(); + unsavedChanges = false; + + return true; +} + +void RecipeInputDialog::saveRecipe( void ) +{ + // Nothing except for the ingredient list (loadedRecipe->ingList) + // was stored before for performance. (recipeID is already there) + + loadedRecipe->photo = sourcePhoto; + loadedRecipe->instructions = instructionsEdit->text(); + loadedRecipe->title = titleEdit->text(); + yieldNumInput->value(loadedRecipe->yield.amount,loadedRecipe->yield.amount_offset); + loadedRecipe->yield.type_id = createNewYieldIfNecessary(yieldTypeEdit->text()); + loadedRecipe->prepTime = prepTimeEdit->time(); + + // Now save() + kdDebug() << "Saving..." << endl; + database->saveRecipe( loadedRecipe ); + + +} + +void RecipeInputDialog::newRecipe( void ) +{ + loadedRecipe->empty(); + TQPixmap image( defaultPhoto ); + photoLabel->setPixmap( image ); + sourcePhoto = TQPixmap(); + + instructionsEdit->setText( i18n( "Write the recipe instructions here" ) ); + instructionsEdit->clearCompletionItems(); + titleEdit->setText( i18n( "Write the recipe title here" ) ); + ingredientList->clear(); + authorShow->clear(); + categoryShow->clear(); + yieldNumInput->setValue( 1, 0 ); + yieldTypeEdit->setText(""); + prepTimeEdit->setTime( TQTime( 0, 0 ) ); + + instructionsEdit->selectAll(); + + //Set back to the first page + tabWidget->setCurrentPage( 0 ); + + ingInput->clear(); + + //Set focus to the title + titleEdit->setFocus(); + titleEdit->selectAll(); + + //clear status info + propertyStatusMapRed.clear(); + propertyStatusMapYellow.clear(); + showStatusIndicator(); +} + +bool RecipeInputDialog::everythingSaved() +{ + return ( !( unsavedChanges ) ); +} + +void RecipeInputDialog::addCategory( void ) +{ + SelectCategoriesDialog *editCategoriesDialog = new SelectCategoriesDialog( this, loadedRecipe->categoryList, database ); + + if ( editCategoriesDialog->exec() == TQDialog::Accepted ) { // user presses Ok + loadedRecipe->categoryList.clear(); + editCategoriesDialog->getSelectedCategories( &( loadedRecipe->categoryList ) ); // get the category list chosen + emit( recipeChanged() ); //Indicate that the recipe changed + + } + + delete editCategoriesDialog; + + // show category list + showCategories(); + + +} + +void RecipeInputDialog::showCategories( void ) +{ + TQString categories; + for ( ElementList::const_iterator cat_it = loadedRecipe->categoryList.begin(); cat_it != loadedRecipe->categoryList.end(); ++cat_it ) { + if ( !categories.isEmpty() ) + categories += ","; + categories += ( *cat_it ).name; + } + categoryShow->setText( categories ); +} + +void RecipeInputDialog::addAuthor( void ) +{ + SelectAuthorsDialog * editAuthorsDialog = new SelectAuthorsDialog( this, loadedRecipe->authorList, database ); + + + if ( editAuthorsDialog->exec() == TQDialog::Accepted ) { // user presses Ok + loadedRecipe->authorList.clear(); + editAuthorsDialog->getSelectedAuthors( &( loadedRecipe->authorList ) ); // get the category list chosen + emit( recipeChanged() ); //Indicate that the recipe changed + } + + delete editAuthorsDialog; + + // show authors list + showAuthors(); +} + +void RecipeInputDialog::showAuthors( void ) +{ + TQString authors; + for ( ElementList::const_iterator author_it = loadedRecipe->authorList.begin(); author_it != loadedRecipe->authorList.end(); ++author_it ) { + if ( !authors.isEmpty() ) + authors += ","; + authors += ( *author_it ).name; + } + authorShow->setText( authors ); +} + +void RecipeInputDialog::enableSaveButton( bool enabled ) +{ + saveButton->setEnabled( enabled ); +} + +void RecipeInputDialog::closeOptions( void ) +{ + + // First check if there's anything unsaved in the recipe + if ( unsavedChanges ) { + + switch ( KMessageBox::questionYesNoCancel( this, i18n( "This recipe contains unsaved changes.\n" "Would you like to save it before closing?" ), i18n( "Unsaved changes" ) ) ) { + case KMessageBox::Yes: + save(); + break; + case KMessageBox::No: + break; + case KMessageBox::Cancel: + return ; + } + + } + + emit enableSaveOption( false ); + unsavedChanges = false; + + // Now close really + emit closeRecipe(); + + +} + +void RecipeInputDialog::showRecipe( void ) +{ + // First check if there's anything unsaved in the recipe + + if ( loadedRecipe->recipeID == -1 ) { + switch ( KMessageBox::questionYesNo( this, i18n( "You need to save the recipe before displaying it. Would you like to save it now?" ), i18n( "Unsaved changes" ) ) ) { + case KMessageBox::Yes: + save(); + break; + case KMessageBox::No: + return ; + } + } + else if ( unsavedChanges ) { + + switch ( KMessageBox::questionYesNoCancel( this, i18n( "This recipe has changes that will not be displayed unless the recipe is saved. Would you like to save it now?" ), i18n( "Unsaved changes" ) ) ) { + case KMessageBox::Yes: + save(); + break; + case KMessageBox::No: + break; + case KMessageBox::Cancel: + return ; + } + + } + + // Now open it really + emit showRecipe( loadedRecipe->recipeID ); +} + +void RecipeInputDialog::spellCheck( void ) +{ + TQString text = instructionsEdit->text(); + KSpellConfig default_cfg( this ); + KSpell::modalCheck( text, &default_cfg ); + KMessageBox::information( this, i18n( "Spell check complete." ) ); + + if ( text != instructionsEdit->text() ) //check if there were changes + instructionsEdit->setText( text ); +} + +void RecipeInputDialog::resizeRecipe( void ) +{ + yieldNumInput->value( loadedRecipe->yield.amount, loadedRecipe->yield.amount_offset ); + ResizeRecipeDialog dlg( this, loadedRecipe ); + + if ( dlg.exec() == TQDialog::Accepted ) + reload(); +} + +int RecipeInputDialog::ingItemIndex( TQListView *listview, const TQListViewItem *item ) const +{ + if ( !item ) + return -1; + + if ( item == listview->firstChild() ) + return 0; + else { + TQListViewItemIterator it( listview->firstChild() ); + uint j = 0; + for ( ; it.current() && it.current() != item; ++it ) { + if ( it.current() ->rtti() == INGLISTVIEWITEM_RTTI ) { + if ( !it.current()->parent() || it.current()->parent()->rtti() == INGGRPLISTVIEWITEM_RTTI ) + j++; + } + } + + if ( !it.current() ) + return -1; + + return j; + } +} + +void RecipeInputDialog::slotIngredientParser() +{ + UnitList units; + database->loadUnits(&units); + IngredientParserDialog dlg(units,this); + if ( dlg.exec() == TQDialog::Accepted ) { + IngredientList ings = dlg.ingredients(); + TQStringList usedGroups; + bool haveHeader = false; + for ( IngredientList::iterator it = ings.begin(); it != ings.end(); ++it ) { + if ( !(*it).group.isEmpty() && usedGroups.find((*it).group) == usedGroups.end() ) { + int id = IngredientInputWidget::createNewGroupIfNecessary((*it).group,database); + addIngredientHeader( Element((*it).group, id) ); + haveHeader = true; + usedGroups << (*it).group; + } + (*it).ingredientID = IngredientInputWidget::createNewIngredientIfNecessary((*it).name,database); + (*it).units.id = IngredientInputWidget::createNewUnitIfNecessary((*it).units.name,false,(*it).ingredientID,(*it).units,database); + + TQValueList<int> prepIDs = IngredientInputWidget::createNewPrepIfNecessary((*it).prepMethodList,database); + TQValueList<int>::const_iterator prep_id_it = prepIDs.begin(); + for ( ElementList::iterator prep_it = (*it).prepMethodList.begin(); prep_it != (*it).prepMethodList.end(); ++prep_it, ++prep_id_it ) { + (*prep_it).id = *prep_id_it; + } + + addIngredient( *it, !haveHeader ); + + if ( usedGroups.count() > 0 && (*it).group.isEmpty() ) { + TQListViewItem *last_item = ingredientList->lastItem(); + if ( last_item->parent() ) { + last_item->parent()->takeItem( last_item ); + ingredientList->insertItem( last_item ); + last_item->moveItem( ingredientList->lastItem()->parent() ); + } + } + } + } +} + +void RecipeInputDialog::slotAddRating() +{ + ElementList criteriaList; + database->loadRatingCriterion(&criteriaList); + + EditRatingDialog ratingDlg(criteriaList,this); + if ( ratingDlg.exec() == TQDialog::Accepted ) { + Rating r = ratingDlg.rating(); + + for ( RatingCriteriaList::iterator rc_it = r.ratingCriteriaList.begin(); rc_it != r.ratingCriteriaList.end(); ++rc_it ) { + int criteria_id = database->findExistingRatingByName((*rc_it).name); + if ( criteria_id == -1 ) { + database->createNewRating((*rc_it).name); + criteria_id = database->lastInsertID(); + } + (*rc_it).id = criteria_id; + } + + RatingDisplayWidget *item = new RatingDisplayWidget; + item->rating_it = loadedRecipe->ratingList.append(r); + addRating(r,item); + ratingListDisplayWidget->insertItem(item,0); + emit( recipeChanged() ); //Indicate that the recipe changed + } +} + +void RecipeInputDialog::addRating( const Rating &rating, RatingDisplayWidget *item ) +{ + int average = tqRound(rating.average()); + if ( average >= 0 ) + item->icon->setPixmap( UserIcon(TQString("rating%1").arg(average) ) ); + else //no rating criteria, therefore no average (we don't want to automatically assume a zero average) + item->icon->clear(); + + item->raterName->setText(rating.rater); + item->comment->setText(rating.comment); + + item->criteriaListView->clear(); + for ( RatingCriteriaList::const_iterator rc_it = rating.ratingCriteriaList.begin(); rc_it != rating.ratingCriteriaList.end(); ++rc_it ) { + TQListViewItem * it = new TQListViewItem(item->criteriaListView,(*rc_it).name); + + int stars = int((*rc_it).stars * 2); //multiply by two to make it easier to work with half-stars + + TQPixmap star = UserIcon(TQString::fromLatin1("star_on")); + int pixmapWidth = 18*(stars/2)+((stars%2==1)?9:0); + TQPixmap generatedPixmap(pixmapWidth,18); + + if ( !generatedPixmap.isNull() ) { //there aren't zero stars + generatedPixmap.fill(); + TQPainter painter( &generatedPixmap ); + painter.drawTiledPixmap(0,0,pixmapWidth,18,star); + it->setPixmap(1,generatedPixmap); + } + } + + item->buttonEdit->disconnect(); + item->buttonRemove->disconnect(); + connect(item->buttonEdit, TQ_SIGNAL(clicked()), + this, TQ_SLOT(slotEditRating())); + connect(item->buttonRemove, TQ_SIGNAL(clicked()), + this, TQ_SLOT(slotRemoveRating())); +} + +void RecipeInputDialog::slotEditRating() +{ + RatingDisplayWidget *sender = (RatingDisplayWidget*)(TQObject::sender()->parent()); + + ElementList criteriaList; + database->loadRatingCriterion(&criteriaList); + + EditRatingDialog ratingDlg(criteriaList,*sender->rating_it,this); + if ( ratingDlg.exec() == TQDialog::Accepted ) { + Rating r = ratingDlg.rating(); + + for ( RatingCriteriaList::iterator rc_it = r.ratingCriteriaList.begin(); rc_it != r.ratingCriteriaList.end(); ++rc_it ) { + int criteria_id = database->findExistingRatingByName((*rc_it).name); + if ( criteria_id == -1 ) { + database->createNewRating((*rc_it).name); + criteria_id = database->lastInsertID(); + } + (*rc_it).id = criteria_id; + } + + (*sender->rating_it) = r; + addRating(r,sender); + emit recipeChanged(); //Indicate that the recipe changed + } +} + +void RecipeInputDialog::slotRemoveRating() +{ + RatingDisplayWidget *sender = (RatingDisplayWidget*)(TQObject::sender()->parent()); + loadedRecipe->ratingList.remove(sender->rating_it); + + //FIXME: sender is removed but never deleted (sender->deleteLater() doesn't work) + ratingListDisplayWidget->removeItem(sender); + + emit recipeChanged(); //Indicate that the recipe changed +} + +void RecipeInputDialog::addIngredient( const Ingredient &ing, bool noHeader ) +{ + Ingredient ingCopy = ing; + + //Append to the ListView + TQListViewItem* lastElement = ingredientList->lastItem(); + while ( lastElement && lastElement->rtti() == INGSUBLISTVIEWITEM_RTTI ) + lastElement = lastElement->itemAbove(); + + if ( noHeader && lastElement ) + lastElement = (lastElement->parent())?lastElement->parent():lastElement; + + if ( !noHeader && lastElement && + ( lastElement->rtti() == INGGRPLISTVIEWITEM_RTTI || ( lastElement->parent() && lastElement->parent() ->rtti() == INGGRPLISTVIEWITEM_RTTI ) ) ) + { + IngGrpListViewItem * header = ( lastElement->parent() ) ? ( IngGrpListViewItem* ) lastElement->parent() : ( IngGrpListViewItem* ) lastElement; + + ingCopy.groupID = header->id(); + + lastElement = new IngListViewItem( header, lastElement, ingCopy ); + for ( TQValueList<IngredientData>::const_iterator it = ingCopy.substitutes.begin(); it != ingCopy.substitutes.end(); ++it ) { + new IngSubListViewItem( lastElement, *it ); + } + lastElement->setOpen(true); + } + else { + lastElement = new IngListViewItem( ingredientList, lastElement, ingCopy ); + for ( TQValueList<IngredientData>::const_iterator it = ing.substitutes.begin(); it != ing.substitutes.end(); ++it ) { + new IngSubListViewItem( lastElement, *it ); + } + lastElement->setOpen(true); + } + + //append to recipe + loadedRecipe->ingList.append( ingCopy ); + + //update the completion in the instructions edit + instructionsEdit->addCompletionItem( ingCopy.name ); + + updatePropertyStatus( ingCopy, true ); + + emit changed(); +} + +void RecipeInputDialog::addIngredientHeader( const Element &header ) +{ + TQListViewItem *last_item = ingredientList->lastItem(); + if ( last_item && last_item->parent() ) + last_item = last_item->parent(); + + IngGrpListViewItem *ing_header = new IngGrpListViewItem( ingredientList, last_item, header.name, header.id ); + ing_header->setOpen( true ); +} + +void RecipeInputDialog::updatePropertyStatus() +{ + propertyStatusMapRed.clear(); + propertyStatusMapYellow.clear(); + + for ( IngredientList::const_iterator ing_it = loadedRecipe->ingList.begin(); ing_it != loadedRecipe->ingList.end(); ++ing_it ) { + updatePropertyStatus( *ing_it, false ); + } + + showStatusIndicator(); +} + +void RecipeInputDialog::updatePropertyStatus( const Ingredient &ing, bool updateIndicator ) +{ + IngredientPropertyList ingPropertyList; + database->loadProperties( &ingPropertyList, ing.ingredientID ); + + if ( ingPropertyList.count() == 0 ) { + propertyStatusMapRed.insert(ing.ingredientID,TQString(i18n("<b>%1:</b> No nutrient information available")).arg(ing.name)); + } + + TQMap<int,bool> ratioCache; //unit->conversion possible + IngredientPropertyList::const_iterator prop_it; + for ( prop_it = ingPropertyList.begin(); prop_it != ingPropertyList.end(); ++prop_it ) { + Ingredient result; + + TQMap<int,bool>::const_iterator cache_it = ratioCache.find((*prop_it).perUnit.id); + if ( cache_it == ratioCache.end() ) { + RecipeDB::ConversionStatus status = database->convertIngredientUnits( ing, (*prop_it).perUnit, result ); + ratioCache.insert((*prop_it).perUnit.id,status==RecipeDB::Success||status==RecipeDB::MismatchedPrepMethod); + + switch ( status ) { + case RecipeDB::Success: break; + case RecipeDB::MissingUnitConversion: { + if ( ing.units.type != Unit::Other && ing.units.type == (*prop_it).perUnit.type ) { + propertyStatusMapRed.insert(ing.ingredientID,TQString(i18n("<b>%3:</b> Unit conversion missing for conversion from '%1' to '%2'")) + .arg(ing.units.name.isEmpty()?i18n("-No unit-"):ing.units.name) + .arg((*prop_it).perUnit.name) + .arg(ing.name)); + } else { + WeightList weights = database->ingredientWeightUnits( ing.ingredientID ); + TQValueList< TQPair<int,int> > usedIds; + TQStringList missingConversions; + for ( WeightList::const_iterator weight_it = weights.begin(); weight_it != weights.end(); ++weight_it ) { + //skip entries that only differ in how it's prepared + TQPair<int,int> usedPair((*weight_it).perAmountUnitID,(*weight_it).weightUnitID); + if ( usedIds.find(usedPair) != usedIds.end() ) + continue; + + TQString toUnit = database->unitName((*weight_it).perAmountUnitID).name; + if ( toUnit.isEmpty() ) toUnit = i18n("-No unit-"); + + TQString fromUnit = database->unitName((*weight_it).weightUnitID).name; + if ( fromUnit.isEmpty() ) fromUnit = i18n("-No unit-"); + + TQString ingUnit = ing.units.name; + if ( ingUnit.isEmpty() ) ingUnit = i18n("-No unit-"); + + TQString propUnit = (*prop_it).perUnit.name; + if ( propUnit.isEmpty() ) propUnit = i18n("-No unit-"); + + missingConversions << conversionPath( ingUnit, toUnit, fromUnit, propUnit); + + usedIds << usedPair; + } + propertyStatusMapRed.insert(ing.ingredientID,TQString(i18n("<b>%1:</b> Either an appropriate ingredient weight entry is needed, or Krecipes needs conversion information to perform one of the following conversions: %2")) + .arg(ing.name) + .arg("<ul><li>"+missingConversions.join("</li><li>")+"</li></ul>") + ); + } + break; + } + case RecipeDB::MissingIngredientWeight: + propertyStatusMapRed.insert(ing.ingredientID,TQString(i18n("<b>%1:</b> No ingredient weight entries")).arg(ing.name)); + break; + case RecipeDB::MismatchedPrepMethod: + if ( ing.prepMethodList.count() == 0 ) + propertyStatusMapRed.insert(ing.ingredientID,TQString(i18n("<b>%1:</b> There is no ingredient weight entry for when no preparation method is specified")).arg(ing.name)); + else + propertyStatusMapRed.insert(ing.ingredientID,TQString(i18n("<b>%1:</b> There is no ingredient weight entry for when prepared in any of the following manners: %2")).arg(ing.name).arg("<ul><li>"+ing.prepMethodList.join("</li><li>")+"</li></ul>")); + break; + case RecipeDB::MismatchedPrepMethodUsingApprox: + propertyStatusMapYellow.insert(ing.ingredientID,TQString(i18n("<b>%1:</b> There is no ingredient weight entry for when prepared in any of the following manners (defaulting to a weight entry without a preparation method specified): %2")).arg(ing.name).arg("<ul><li>"+ing.prepMethodList.join("</li><li>")+"</li></ul>")); + break; + default: kdDebug()<<"Code error: Unhandled conversion status code "<<status<<endl; break; + } + } + } + + if ( updateIndicator ) + showStatusIndicator(); +} + +void RecipeInputDialog::showStatusIndicator() +{ + if ( propertyStatusMapRed.count() == 0 ) { + if ( propertyStatusMapYellow.count() == 0 ) { + propertyStatusLed->setColor( TQt::green ); + propertyStatusLabel->setText( i18n("Complete") ); + propertyStatusButton->setEnabled(false); + } + else { + propertyStatusLed->setColor( TQt::yellow ); + propertyStatusLabel->setText( i18n("Complete, but approximations made") ); + propertyStatusButton->setEnabled(true); + } + } + else { + propertyStatusLed->setColor( TQt::red ); + propertyStatusLabel->setText( i18n("Incomplete") ); + propertyStatusButton->setEnabled(true); + } + + if ( propertyStatusMapRed.count() == 0 && propertyStatusMapYellow.count() == 0 ) + propertyStatusDialog->hide(); + else + statusTextView->setText(statusMessage()); +} + +TQString RecipeInputDialog::statusMessage() const +{ + TQString statusMessage; + + if ( propertyStatusMapRed.count() > 0 ) { + statusMessage.append( i18n("The nutrient information for this recipe is incomplete because the following information is missing:") ); + statusMessage.append("<ul>"); + for ( TQMap<int,TQString>::const_iterator it = propertyStatusMapRed.begin(); it != propertyStatusMapRed.end(); ++it ) { + statusMessage.append("<li>"); + statusMessage.append(it.data()); + statusMessage.append("</li>"); + } + statusMessage.append("</ul>"); + } + + if ( propertyStatusMapYellow.count() > 0 ) { + statusMessage.append( i18n("The following approximations will be made when determining nutrient information:") ); + statusMessage.append("<ul>"); + for ( TQMap<int,TQString>::const_iterator it = propertyStatusMapYellow.begin(); it != propertyStatusMapYellow.end(); ++it ) { + statusMessage.append("<li>"); + statusMessage.append(it.data()); + statusMessage.append("</li>"); + } + statusMessage.append("</ul>"); + } + + return statusMessage; +} + +TQString RecipeInputDialog::conversionPath( const TQString &ingUnit, const TQString &toUnit, const TQString &fromUnit, const TQString &propUnit ) const +{ + TQString path = "'"+ingUnit+"'"; + + TQString lastUnit = ingUnit; + if ( lastUnit != toUnit ) { + path.append(" => '"+toUnit+"'"); + lastUnit = toUnit; + } + if ( lastUnit != fromUnit ) { + path.append(" => '"+fromUnit+"'"); + lastUnit = fromUnit; + } + if ( lastUnit != propUnit ) { + path.append(" => '"+propUnit+"'"); + lastUnit = propUnit; + } + return path; +} + +#include "recipeinputdialog.moc" diff --git a/src/dialogs/recipeinputdialog.h b/src/dialogs/recipeinputdialog.h new file mode 100644 index 0000000..dd9a623 --- /dev/null +++ b/src/dialogs/recipeinputdialog.h @@ -0,0 +1,228 @@ +/*************************************************************************** +* Copyright (C) 2003-2005 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef RECIPEINPUTDIALOG_H +#define RECIPEINPUTDIALOG_H + +#include <kdialog.h> +#include <ktextedit.h> +#include <klineedit.h> +#include <kcombobox.h> +#include <tdelistview.h> +#include <knuminput.h> +#include <kpushbutton.h> +#include <kiconloader.h> +#include <kled.h> + +#include <tqlabel.h> +#include <tqgroupbox.h> +#include <tqmap.h> +#include <tqobject.h> +#include <tqtabwidget.h> +#include <tqtoolbutton.h> +#include <tqvbox.h> + +#include "datablocks/elementlist.h" + +class TQTabWidget; +class TQTimeEdit; +class TQDragEvent; +class TQButtonGroup; +class TQWidgetStack; +class TQTextEdit; + +class KreTextEdit; +class KWidgetListbox; +class KDialogBase; + +class ImageDropLabel; +class Recipe; +class ElementList; +class RecipeDB; +class FractionInput; +class Ingredient; +class Rating; +class RatingDisplayWidget; +class IngredientInputWidget; +class ClickableLed; + +/** +@author Unai Garro +*/ +class RecipeInputDialog: public TQVBox +{ + TQ_OBJECT + +public: + + RecipeInputDialog( TQWidget* parent, RecipeDB* db ); + void loadRecipe( int recipeID ); + ~RecipeInputDialog(); + void newRecipe( void ); + bool everythingSaved(); + void reload( void ); + int loadedRecipeID() const; + +private: + + // Internal Data + Recipe *loadedRecipe; //Loaded Recipe + RecipeDB *database; + bool changedSignalEnabled; + bool unsavedChanges; + + // Widgets + TQTabWidget* tabWidget; + TQGroupBox* recipeTab; + TQGroupBox* instructionsTab; + + //Recipe Photo + ImageDropLabel *photoLabel; + TQPixmap sourcePhoto; + TQPushButton *changePhotoButton; + + //Recipe Body + KreTextEdit* instructionsEdit; + TQLabel* titleLabel; + KLineEdit* titleEdit; + + //Additional recipe data + TQLabel* yieldLabel; + FractionInput* yieldNumInput; + KLineEdit* yieldTypeEdit; + TQTimeEdit *prepTimeEdit; + TQLabel* authorLabel; + KLineEdit* authorShow; + TQPushButton* addAuthorButton; + TQLabel* categoryLabel; + KLineEdit* categoryShow; + TQPushButton* addCategoryButton; + + //Ingredient inputs + TDEListView* ingredientList; + TQGroupBox *ingredientGBox; + IngredientInputWidget *ingInput; + ClickableLed *propertyStatusLed; + TQLabel *propertyStatusLabel; + TQPushButton *propertyStatusButton; + TQTextEdit *statusTextView; + KDialogBase *propertyStatusDialog; + + // Buttons to move ingredients up & down... + KPushButton* upButton; + KPushButton* downButton; + KPushButton* removeButton; + KPushButton* addButton; + KPushButton* ingParserButton; + + //Function buttons + TQGroupBox* functionsBox; + TQToolButton* saveButton; + TQToolButton* closeButton; + TQToolButton* showButton; + TQToolButton* resizeButton; + + TQToolButton* spellCheckButton; + + KWidgetListbox *ratingListDisplayWidget; + + TQMap<int,TQString> propertyStatusMapRed; + TQMap<int,TQString> propertyStatusMapYellow; + + // Internal functions + int createNewYieldIfNecessary( const TQString &yield ); + void saveRecipe( void ); + void showCategories( void ); + void showAuthors( void ); + int ingItemIndex( TQListView *listview, const TQListViewItem *item ) const; + void addRating( const Rating &rating, RatingDisplayWidget *item ); + TQString statusMessage() const; + TQString conversionPath( const TQString &ingUnit, const TQString &toUnit, const TQString &fromUnit, const TQString &propUnit ) const; + + // Signals & Slots + +private slots: + void changePhoto( void ); + void clearPhoto( void ); + void moveIngredientUp( void ); + void moveIngredientDown( void ); + void removeIngredient( void ); + void syncListView( TQListViewItem* it, const TQString &new_text, int col ); + void recipeChanged( void ); + void recipeChanged( const TQString &t ); + void enableChangedSignal( bool en = true ); + void addCategory( void ); + void addAuthor( void ); + void enableSaveButton( bool enabled ); + void closeOptions( void ); + void showRecipe( void ); + void prepTitleChanged( const TQString &title ); + void recipeRemoved( int id ); + void slotIngredientParser(); + void slotAddRating(); + void slotEditRating(); + void slotRemoveRating(); + void addIngredient( const Ingredient &ing, bool noHeader = false ); + void addIngredientHeader( const Element &header ); + void updatePropertyStatus(); + void updatePropertyStatus( const Ingredient &ing, bool updateIndicator ); + void showStatusIndicator(); + +public slots: + bool save ( void ); // Activated when krecipes.cpp sends signal save() + void spellCheck( void ); + void resizeRecipe( void ); + +signals: + void changed( void ); + void closeRecipe( void ); + void createButton( TQWidget* w, const TQString &title ); + void enableSaveOption( bool en = true ); + void showRecipe( int recipeID ); //Indicates krecipesview to show it + void titleChanged( const TQString &title ); + + +}; + +class ClickableLed : public KLed +{ +TQ_OBJECT + +public: + ClickableLed( TQWidget *parent ); + +protected: + virtual void mouseReleaseEvent( TQMouseEvent* ); + +signals: + void clicked(); +}; + +class ImageDropLabel : public TQLabel +{ + TQ_OBJECT + +public: + ImageDropLabel( TQWidget *parent, TQPixmap &_sourcePhoto ); + +signals: + void changed(); + +protected: + void dragEnterEvent( TQDragEnterEvent* event ); + void dropEvent( TQDropEvent* event ); + +private: + TQPixmap &sourcePhoto; +}; + +#endif diff --git a/src/dialogs/recipeprintpreview.cpp b/src/dialogs/recipeprintpreview.cpp new file mode 100644 index 0000000..7350d11 --- /dev/null +++ b/src/dialogs/recipeprintpreview.cpp @@ -0,0 +1,62 @@ +/*************************************************************************** +* Copyright (C) 2006 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "recipeprintpreview.h" + +#include <tqvbox.h> + +#include <kdebug.h> +#include <tdelocale.h> + +#include "recipeviewdialog.h" +#include "pagesetupdialog.h" + +RecipePrintPreview::RecipePrintPreview( TQWidget *parent, RecipeDB *db, const TQValueList<int> &ids ) + : KDialogBase( parent, "RecipePrintPreview", true, i18n("Print Preview"), + KDialogBase::Ok | KDialogBase::Cancel | KDialogBase::User1 | KDialogBase::Help, KDialogBase::Ok ) +{ + setHelp("print-recipe"); + setButtonText( KDialogBase::User1, i18n("&Edit") ); + setButtonText( KDialogBase::Ok, i18n("&Print") ); + + setSizeGripEnabled( true ); + + // Initialize UI Elements + TQVBox *page = makeVBoxMainWidget(); + + recipeView = new RecipeViewDialog( page, db ); + recipeView->loadRecipes( ids, "Print" ); + + setInitialSize( TQSize(450,500), false ); +} + +RecipePrintPreview::~RecipePrintPreview() +{ +} + +void RecipePrintPreview::slotOk() +{ + recipeView->printNoPreview(); + accept(); +} + +void RecipePrintPreview::slotUser1( void ) +{ + PageSetupDialog pageSetup( this, Recipe(), "Print" ); + if ( pageSetup.exec() == TQDialog::Accepted ) + reload(); +} + +void RecipePrintPreview::reload() +{ + recipeView->reload( "Print" ); +} + +#include "recipeprintpreview.moc" diff --git a/src/dialogs/recipeprintpreview.h b/src/dialogs/recipeprintpreview.h new file mode 100644 index 0000000..f4c7b65 --- /dev/null +++ b/src/dialogs/recipeprintpreview.h @@ -0,0 +1,41 @@ +/*************************************************************************** +* Copyright (C) 2006 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef RECIPEPRINTPREVIEW_H +#define RECIPEPRINTPREVIEW_H + +#include <tqstring.h> +#include <tqvaluelist.h> + +#include <kdialogbase.h> + +class RecipeDB; +class RecipeViewDialog; + +class RecipePrintPreview : public KDialogBase +{ +TQ_OBJECT + +public: + RecipePrintPreview( TQWidget *parent, RecipeDB *db, const TQValueList<int> &ids ); + ~RecipePrintPreview(); + + void reload(); + +public slots: + void slotOk(); + void slotUser1( void ); + +private: + // Internal Variables + RecipeViewDialog *recipeView; +}; + +#endif diff --git a/src/dialogs/recipeviewdialog.cpp b/src/dialogs/recipeviewdialog.cpp new file mode 100644 index 0000000..133aece --- /dev/null +++ b/src/dialogs/recipeviewdialog.cpp @@ -0,0 +1,167 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "recipeviewdialog.h" + +#include <tqlayout.h> +#include <tqstyle.h> +#include <tqfile.h> + +#include <tdeapplication.h> +#include <kcursor.h> +#include <kdebug.h> +#include <tdehtmlview.h> +#include <tdehtml_part.h> +#include <tdelocale.h> +#include <tdemainwindow.h> +#include <kprogress.h> +#include <kstandarddirs.h> +#include <kstatusbar.h> +#include <tdeconfig.h> +#include <tdeglobal.h> + +#include "datablocks/mixednumber.h" +#include "backends/recipedb.h" +#include "exporters/htmlexporter.h" +#include "recipeprintpreview.h" + +RecipeViewDialog::RecipeViewDialog( TQWidget *parent, RecipeDB *db, int recipeID ) : TQVBox( parent ), + database(db) +{ + // Initialize UI Elements + recipeView = new TDEHTMLPart( this ); + + connect( database, TQ_SIGNAL(recipeRemoved(int)), TQ_SLOT(recipeRemoved(int)) ); + + tmp_filename = locateLocal( "tmp", "krecipes_recipe_view" ); + + //----------Load the recipe -------- + if ( recipeID != -1 ) + loadRecipe( recipeID ); +} + +RecipeViewDialog::~RecipeViewDialog() +{ + if ( recipe_loaded ) + removeOldFiles(); +} + +bool RecipeViewDialog::loadRecipe( int recipeID ) +{ + TQValueList<int> ids; + ids.append( recipeID ); + return loadRecipes( ids ); +} + +bool RecipeViewDialog::loadRecipes( const TQValueList<int> &ids, const TQString &layoutConfig ) +{ + TDEApplication::setOverrideCursor( KCursor::waitCursor() ); + + // Remove any files created by the last recipe loaded + removeOldFiles(); + + ids_loaded = ids; //need to save these ids in order to delete the html files later...make sure this comes after the call to removeOldFiles() + recipe_loaded = ( ids.count() > 0 && ids[ 0 ] >= 0 ); + + bool success = showRecipes( ids, layoutConfig ); + + TDEApplication::restoreOverrideCursor(); + return success; +} + +bool RecipeViewDialog::showRecipes( const TQValueList<int> &ids, const TQString &layoutConfig ) +{ + KProgressDialog * progress_dialog = 0; + + if ( ids.count() > 1 ) //we don't want a progress bar coming up when there is only one recipe... it may show up during the splash screen + { + progress_dialog = new KProgressDialog( this, "open_progress_dialog", TQString::null, i18n( "Opening recipes, please wait..." ), true ); + progress_dialog->resize( 240, 80 ); + } + + HTMLExporter html_generator( tmp_filename + ".html", "html" ); + if ( layoutConfig != TQString::null ) { + TDEConfig *config = TDEGlobal::config(); + config->setGroup( "Page Setup" ); + TQString styleFile = config->readEntry( layoutConfig+"Layout", locate( "appdata", "layouts/Default.klo" ) ); + if ( !styleFile.isEmpty() && TQFile::exists( styleFile ) ) + html_generator.setStyle( styleFile ); + + TQString templateFile = config->readEntry( layoutConfig+"Template", locate( "appdata", "layouts/Default.template" ) ); + if ( !templateFile.isEmpty() && TQFile::exists( templateFile ) ) + html_generator.setTemplate( templateFile ); + } + + html_generator.exporter( ids, database, progress_dialog ); //writes the generated HTML to 'tmp_filename+".html"' + if ( progress_dialog && progress_dialog->wasCancelled() ) { + delete progress_dialog; + return false; + } + + delete recipeView; // Temporary workaround + recipeView = new TDEHTMLPart( this ); // to avoid the problem of caching images of TDEHTMLPart + + KURL url; + url.setPath( tmp_filename + ".html" ); + recipeView->openURL( url ); + recipeView->show(); + kdDebug() << "Opening URL: " << url.htmlURL() << endl; + + delete progress_dialog; + return true; +} + +void RecipeViewDialog::print() +{ + if ( recipe_loaded ) { + RecipePrintPreview preview( this, database, ids_loaded ); + preview.exec(); + } +} + +void RecipeViewDialog::printNoPreview( void ) +{ + if ( recipe_loaded ) { + recipeView->view() ->print(); + } +} + +void RecipeViewDialog::reload( const TQString &layoutConfig ) +{ + loadRecipes( ids_loaded, layoutConfig ); +} + +void RecipeViewDialog::removeOldFiles() +{ + if ( ids_loaded.count() > 0 ) { + RecipeList recipe_list; + database->loadRecipes( &recipe_list, RecipeDB::Title, ids_loaded ); + + TQValueList<int> recipe_ids; + for ( RecipeList::const_iterator it = recipe_list.begin(); it != recipe_list.end(); ++it ) + recipe_ids << ( *it ).recipeID; + + HTMLExporter::removeHTMLFiles( tmp_filename, recipe_ids ); + } +} + +void RecipeViewDialog::recipeRemoved( int id ) +{ + //if the deleted recipe is loaded, clean the view up + if ( ids_loaded.find(id) != ids_loaded.end() ) { + Recipe recipe; database->loadRecipe( &recipe, RecipeDB::Title, id ); + HTMLExporter::removeHTMLFiles( tmp_filename, recipe.recipeID ); + ids_loaded.remove(id); + } +} + +#include "recipeviewdialog.moc" diff --git a/src/dialogs/recipeviewdialog.h b/src/dialogs/recipeviewdialog.h new file mode 100644 index 0000000..2b7c884 --- /dev/null +++ b/src/dialogs/recipeviewdialog.h @@ -0,0 +1,77 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef RECIPEVIEWDIALOG_H +#define RECIPEVIEWDIALOG_H + +#include <tqvbox.h> +#include <tqstring.h> + +class RecipeDB; +class Recipe; +class TDEHTMLPart; +class TQPushButton; + +/** +@author Unai Garro +*/ + +class RecipeViewDialog : public TQVBox +{ + TQ_OBJECT + +public: + RecipeViewDialog( TQWidget *parent, RecipeDB *db, int recipeID = -1 ); + + ~RecipeViewDialog(); + + /** @return Boolean indicating whether or not the recipe was successfully loaded */ + bool loadRecipe( int recipeID ); + + /** @return Boolean indicating whether or not the recipes were successfully loaded */ + bool loadRecipes( const TQValueList<int> &ids, const TQString &layoutConfig = TQString::null ); + + int recipesLoaded() const + { + return ids_loaded.count(); + } + const TQValueList<int> currentRecipes() const + { + return ids_loaded; + } + + void reload( const TQString &layoutConfig = TQString::null ); + +public slots: + void printNoPreview( void ); + void print( void ); + +private: + + // Internal Variables + TDEHTMLPart *recipeView; + RecipeDB *database; + bool recipe_loaded; + TQValueList<int> ids_loaded; + TQString tmp_filename; + + // Internal Methods + bool showRecipes( const TQValueList<int> &ids, const TQString &layoutConfig ); + void removeOldFiles(); + +private slots: + void recipeRemoved(int); + +}; + + +#endif diff --git a/src/dialogs/refineshoppinglistdialog.cpp b/src/dialogs/refineshoppinglistdialog.cpp new file mode 100644 index 0000000..adbe5bb --- /dev/null +++ b/src/dialogs/refineshoppinglistdialog.cpp @@ -0,0 +1,206 @@ +/*************************************************************************** +* Copyright (C) 2004 by Jason Kivlighn * +* ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "refineshoppinglistdialog.h" + +#include <tqvariant.h> +#include <tqpushbutton.h> +#include <tqlabel.h> +#include <tqheader.h> +#include <tqlayout.h> +#include <tqtooltip.h> +#include <tqwhatsthis.h> + +#include <tdelocale.h> +#include <kiconloader.h> +#include <tdeapplication.h> +#include <kcursor.h> +#include <tdeconfig.h> + +#include "backends/recipedb.h" +#include "widgets/krelistview.h" +#include "widgets/ingredientlistview.h" +#include "shoppinglistviewdialog.h" +#include "shoppingcalculator.h" +#include "datablocks/mixednumber.h" + +RefineShoppingListDialog::RefineShoppingListDialog( TQWidget* parent, RecipeDB *db, const ElementList &recipeList ) + : KDialogBase( parent, "refinedialog", true, TQString::null, + KDialogBase::Ok, KDialogBase::Ok ), + database( db ) +{ + setButtonText( KDialogBase::Ok, i18n( "&Done" ) ); + + TQVBox *page = makeVBoxMainWidget(); + + helpLabel = new TQLabel( page, "helpLabel" ); + helpLabel->setTextFormat( TQLabel::RichText ); + + TQWidget *layout2Widget = new TQWidget(page); + + TQHBoxLayout *layout2 = new TQHBoxLayout( layout2Widget, 0, 6, "layout2" ); + + allIngListView = new KreListView( layout2Widget, TQString::null, true, 0 ); + StdIngredientListView *list_view = new StdIngredientListView(allIngListView,database); + list_view->reload(); + allIngListView->setListView(list_view); + layout2->addWidget( allIngListView ); + + layout1 = new TQVBoxLayout( 0, 0, 6, "layout1" ); + + TDEIconLoader il; + + addButton = new TQPushButton( layout2Widget, "addButton" ); + addButton->setIconSet( il.loadIconSet( "forward", TDEIcon::Small ) ); + addButton->setFixedSize( TQSize( 32, 32 ) ); + layout1->addWidget( addButton ); + + removeButton = new TQPushButton( layout2Widget, "removeButton" ); + removeButton->setIconSet( il.loadIconSet( "back", TDEIcon::Small ) ); + removeButton->setFixedSize( TQSize( 32, 32 ) ); + layout1->addWidget( removeButton ); + spacer1 = new TQSpacerItem( 51, 191, TQSizePolicy::Minimum, TQSizePolicy::Expanding ); + layout1->addItem( spacer1 ); + layout2->addLayout( layout1 ); + + ingListView = new KreListView( layout2Widget, TQString::null, true ); + ingListView->listView() ->addColumn( i18n( "Ingredients in Shopping List" ) ); + ingListView->listView() ->addColumn( i18n( "Amount" ) ); + ingListView->listView() ->addColumn( i18n( "Unit" ) ); + ingListView->listView() ->setItemsRenameable( true ); + ingListView->listView() ->setRenameable( 0, false ); + ingListView->listView() ->setRenameable( 1, true ); + ingListView->listView() ->setRenameable( 2, true ); + layout2->addWidget( ingListView ); + + languageChange(); + + clearWState( WState_Polished ); + + connect( addButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( addIngredient() ) ); + connect( removeButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( removeIngredient() ) ); + connect( ingListView->listView(), TQ_SIGNAL( itemRenamed( TQListViewItem*, const TQString &, int ) ), TQ_SLOT( itemRenamed( TQListViewItem*, const TQString &, int ) ) ); + + TDEApplication::setOverrideCursor( KCursor::waitCursor() ); + calculateShopping( recipeList, &ingredientList, database ); + TDEApplication::restoreOverrideCursor(); + + loadData(); +} + +RefineShoppingListDialog::~RefineShoppingListDialog() +{} + +void RefineShoppingListDialog::languageChange() +{ + helpLabel->setText( i18n( "On the right are the ingredients needed for the recipes you selected. You may now add additional ingredients, remove ingredients you do not need, or modify the amounts of existing ingredients." ) ); + allIngListView->listView() ->header() ->setLabel( 0, i18n( "Ingredients" ) ); + ingListView->listView() ->header() ->setLabel( 0, i18n( "Ingredients in Shopping List" ) ); + ingListView->listView() ->header() ->setLabel( 1, i18n( "Amount" ) ); + ingListView->listView() ->header() ->setLabel( 2, i18n( "Unit" ) ); +} + +void RefineShoppingListDialog::accept() +{ + hide(); + + ShoppingListViewDialog view( this, ingredientList ); + view.exec(); + + TQDialog::accept(); +} + +void RefineShoppingListDialog::loadData() +{ + for ( IngredientList::iterator it = ingredientList.begin(); it != ingredientList.end(); ++it ) { + //from here on, the shopping list will work with the upper value of the range (if exists) + (*it).amount = (*it).amount+(*it).amount_offset; + (*it).amount_offset = 0; + + TQString amount_str; + if ( ( *it ).amount > 0 ) { + TDEConfig * config = kapp->config(); + config->setGroup( "Formatting" ); + + if ( config->readBoolEntry( "Fraction" ) ) + amount_str = MixedNumber( ( *it ).amount ).toString(); + else + amount_str = beautify( TDEGlobal::locale() ->formatNumber( ( *it ).amount, 5 ) ); + } + + TQListViewItem *new_item = new TQListViewItem( ingListView->listView(), ( *it ).name, amount_str, ( *it ).amount > 1 ? ( *it ).units.plural : ( *it ).units.name ); + item_ing_map.insert( new_item, it ); + } +} + +void RefineShoppingListDialog::addIngredient() +{ + TQListViewItem * item = allIngListView->listView() ->selectedItem(); + if ( item ) { + TQListViewItem * new_item = new TQListViewItem( ingListView->listView(), item->text( 0 ) ); + ingListView->listView() ->setSelected( new_item, true ); + ingListView->listView() ->ensureItemVisible( new_item ); + allIngListView->listView() ->setSelected( item, false ); + + item_ing_map.insert( new_item, ingredientList.append( Ingredient( item->text( 0 ), 0, Unit() ) ) ); + } +} + +void RefineShoppingListDialog::removeIngredient() +{ + TQListViewItem * item = ingListView->listView() ->selectedItem(); + if ( item ) { + for ( IngredientList::iterator it = ingredientList.begin(); it != ingredientList.end(); ++it ) { + if ( *item_ing_map.find( item ) == it ) { + ingredientList.remove( it ); + item_ing_map.remove( item ); + break; + } + } + delete item; + } +} + +void RefineShoppingListDialog::itemRenamed( TQListViewItem* item, const TQString &new_text, int col ) +{ + if ( col == 1 ) { + IngredientList::iterator found_it = *item_ing_map.find( item ); + + bool ok; + MixedNumber amount = MixedNumber::fromString( new_text, &ok ); + if ( ok ) { + ( *found_it ).amount = amount.toDouble(); + } + else { //revert back to the valid amount string + TQString amount_str; + if ( ( *found_it ).amount > 0 ) { + TDEConfig * config = kapp->config(); + config->setGroup( "Formatting" ); + + if ( config->readBoolEntry( "Fraction" ) ) + amount_str = MixedNumber( ( *found_it ).amount ).toString(); + else + amount_str = beautify( TDEGlobal::locale() ->formatNumber( ( *found_it ).amount, 5 ) ); + } + + item->setText( 1, amount_str ); + } + } + else if ( col == 2 ) { + IngredientList::iterator found_it = *item_ing_map.find( item ); + + if ( ( *found_it ).amount > 1 ) + ( *found_it ).units.plural = new_text; + else + ( *found_it ).units.name = new_text; + } +} + +#include "refineshoppinglistdialog.moc" diff --git a/src/dialogs/refineshoppinglistdialog.h b/src/dialogs/refineshoppinglistdialog.h new file mode 100644 index 0000000..5952a9b --- /dev/null +++ b/src/dialogs/refineshoppinglistdialog.h @@ -0,0 +1,66 @@ +/*************************************************************************** +* Copyright (C) 2004 by Jason Kivlighn * +* ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef REFINESHOPPINGLISTDIALOG_H +#define REFINESHOPPINGLISTDIALOG_H + +#include <tqvariant.h> +#include <tqmap.h> + +#include <kdialogbase.h> + +#include "datablocks/ingredientlist.h" + +class TQVBoxLayout; +class TQHBoxLayout; +class TQGridLayout; +class TQSpacerItem; +class TQLabel; +class TQListViewItem; +class TQPushButton; + +class RecipeDB; +class ElementList; +class KreListView; + +class RefineShoppingListDialog : public KDialogBase +{ + TQ_OBJECT + +public: + RefineShoppingListDialog( TQWidget* parent, RecipeDB *db, const ElementList &recipeList ); + ~RefineShoppingListDialog(); + + TQLabel* helpLabel; + KreListView* allIngListView; + TQPushButton* addButton; + TQPushButton* removeButton; + KreListView* ingListView; + +protected: + TQVBoxLayout* layout1; + TQSpacerItem* spacer1; + +protected slots: + virtual void languageChange(); + virtual void accept(); + void addIngredient(); + void removeIngredient(); + void itemRenamed( TQListViewItem*, const TQString &, int ); + +private: + void loadData(); + + RecipeDB *database; + IngredientList ingredientList; + TQMap<TQListViewItem*, IngredientList::iterator> item_ing_map; +}; + +#endif // REFINESHOPPINGLISTDIALOG_H diff --git a/src/dialogs/resizerecipedialog.cpp b/src/dialogs/resizerecipedialog.cpp new file mode 100644 index 0000000..958fadb --- /dev/null +++ b/src/dialogs/resizerecipedialog.cpp @@ -0,0 +1,191 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "resizerecipedialog.h" + +#include <cmath> + +#include <tqvbox.h> +#include <tqvariant.h> +#include <tqbuttongroup.h> +#include <tqframe.h> +#include <tqlabel.h> +#include <knuminput.h> +#include <klineedit.h> +#include <tqradiobutton.h> +#include <tqlayout.h> +#include <tqtooltip.h> +#include <tqwhatsthis.h> +#include <tqimage.h> +#include <tqpixmap.h> + +#include <tdelocale.h> +#include <tdemessagebox.h> +#include <kdebug.h> + +#include "datablocks/recipe.h" +#include "widgets/fractioninput.h" + +#define FACTOR_RADIO_BUTTON 0 +#define SERVINGS_RADIO_BUTTON 1 + +ResizeRecipeDialog::ResizeRecipeDialog( TQWidget *parent, Recipe *recipe ) + : KDialogBase( parent, "ResizeRecipeDialog", true, i18n( "Resize Recipe" ), + KDialogBase::Ok | KDialogBase::Cancel, KDialogBase::Ok ), + m_recipe( recipe ) +{ + TQVBox *page = makeVBoxMainWidget(); + + buttonGroup = new TQButtonGroup( page ); + buttonGroup->setSizePolicy( TQSizePolicy( ( TQSizePolicy::SizeType ) 5, ( TQSizePolicy::SizeType ) 7, 0, 1, buttonGroup->sizePolicy().hasHeightForWidth() ) ); + buttonGroup->setLineWidth( 0 ); + buttonGroup->setColumnLayout( 0, TQt::Vertical ); + buttonGroup->layout() ->setSpacing( 6 ); + buttonGroup->layout() ->setMargin( 11 ); + buttonGroupLayout = new TQVBoxLayout( buttonGroup->layout() ); + buttonGroupLayout->setAlignment( TQt::AlignTop ); + + yieldRadioButton = new TQRadioButton( buttonGroup ); + buttonGroup->insert( yieldRadioButton, SERVINGS_RADIO_BUTTON ); + buttonGroupLayout->addWidget( yieldRadioButton ); + + yieldFrame = new TQFrame( buttonGroup ); + yieldFrame->setFrameShape( TQFrame::Box ); + yieldFrame->setFrameShadow( TQFrame::Sunken ); + yieldFrame->setLineWidth( 1 ); + yieldFrameLayout = new TQGridLayout( yieldFrame, 1, 1, 11, 6 ); + + currentYieldLabel = new TQLabel( yieldFrame ); + + yieldFrameLayout->addWidget( currentYieldLabel, 0, 0 ); + + newYieldLabel = new TQLabel( yieldFrame ); + + yieldFrameLayout->addMultiCellWidget( newYieldLabel, 1, 1, 0, 1 ); + + currentYieldInput = new KLineEdit( yieldFrame ); + currentYieldInput->setReadOnly( TRUE ); + currentYieldInput->setAlignment( TQt::AlignRight ); + yieldFrameLayout->addMultiCellWidget( currentYieldInput, 0, 0, 1, 2 ); + + newYieldInput = new FractionInput( yieldFrame ); + yieldFrameLayout->addWidget( newYieldInput, 1, 2 ); + + buttonGroupLayout->addWidget( yieldFrame ); + + factorRadioButton = new TQRadioButton( buttonGroup ); + buttonGroup->insert( factorRadioButton, FACTOR_RADIO_BUTTON ); + buttonGroupLayout->addWidget( factorRadioButton ); + + factorFrame = new TQFrame( buttonGroup ); + factorFrame->setSizePolicy( TQSizePolicy( ( TQSizePolicy::SizeType ) 7, ( TQSizePolicy::SizeType ) 5, 1, 0, factorFrame->sizePolicy().hasHeightForWidth() ) ); + factorFrame->setFrameShape( TQFrame::Box ); + factorFrame->setFrameShadow( TQFrame::Sunken ); + factorFrame->setLineWidth( 1 ); + factorFrameLayout = new TQHBoxLayout( factorFrame, 11, 6 ); + + factorLabel = new TQLabel( factorFrame ); + factorFrameLayout->addWidget( factorLabel ); + + factorInput = new FractionInput( factorFrame ); + factorInput->setSizePolicy( TQSizePolicy( ( TQSizePolicy::SizeType ) 7, ( TQSizePolicy::SizeType ) 5, 0, 0, factorInput->sizePolicy().hasHeightForWidth() ) ); + factorFrameLayout->addWidget( factorInput ); + buttonGroupLayout->addWidget( factorFrame ); + + languageChange(); + + + newYieldInput->setValue( m_recipe->yield.amount, 0 ); //Ignore the range info, it doesn't work in this context + currentYieldInput->setText( m_recipe->yield.toString() ); + + if ( recipe->yield.amount_offset > 0 ) { + yieldRadioButton->setEnabled(false); + buttonGroup->setButton( FACTOR_RADIO_BUTTON ); + activateCurrentOption( FACTOR_RADIO_BUTTON ); + } + else { + buttonGroup->setButton( SERVINGS_RADIO_BUTTON ); + activateCurrentOption( SERVINGS_RADIO_BUTTON ); + } + + // signals and slots connections + connect( buttonGroup, TQ_SIGNAL( clicked( int ) ), this, TQ_SLOT( activateCurrentOption( int ) ) ); +} + +void ResizeRecipeDialog::languageChange() +{ + buttonGroup->setTitle( TQString::null ); + yieldRadioButton->setText( i18n( "Scale by yield" ) ); + newYieldLabel->setText( i18n( "New yield:" ) ); + currentYieldLabel->setText( i18n( "Current yield:" ) ); + factorRadioButton->setText( i18n( "Scale by factor" ) ); + factorLabel->setText( i18n( "Factor (i.e. 1/2 to half, 3 to triple):" ) ); +} + +void ResizeRecipeDialog::activateCurrentOption( int button_id ) +{ + switch ( button_id ) { + case SERVINGS_RADIO_BUTTON: + factorFrame->setEnabled( false ); + yieldFrame->setEnabled( true ); + break; + case FACTOR_RADIO_BUTTON: + factorFrame->setEnabled( true ); + yieldFrame->setEnabled( false ); + break; + default: + break; + } +} + +void ResizeRecipeDialog::accept() +{ + if ( currentYieldInput->text().toInt() == 0 ) + KMessageBox::error( this, i18n( "Unable to scale a recipe with zero yield" ) ); + else if ( buttonGroup->selected() == yieldRadioButton ) { + if ( newYieldInput->isInputValid() ) { + double new_yield = newYieldInput->value().toDouble(); + double current_yield = MixedNumber::fromString(currentYieldInput->text()).toDouble(); + + resizeRecipe( new_yield / current_yield ); + } + else { + KMessageBox::error( this, i18n( "Invalid input" ) ); + newYieldInput->selectAll(); + return; + } + } + else { + if ( factorInput->isInputValid() && factorInput->value() > 0 ) + resizeRecipe( factorInput->value().toDouble() ); + else { + KMessageBox::error( this, i18n( "Invalid input" ) ); + factorInput->selectAll(); + return ; + } + } + + TQDialog::accept(); +} + +//TODO YIELD: handle ranges +void ResizeRecipeDialog::resizeRecipe( double factor ) +{ + m_recipe->yield.amount = MixedNumber::fromString(currentYieldInput->text()).toDouble() * factor; + + for ( IngredientList::iterator ing_it = m_recipe->ingList.begin(); ing_it != m_recipe->ingList.end(); ++ing_it ) { + ( *ing_it ).amount = ( *ing_it ).amount * factor; + ( *ing_it ).amount_offset = ( *ing_it ).amount_offset * factor; + } +} + +#include "resizerecipedialog.moc" diff --git a/src/dialogs/resizerecipedialog.h b/src/dialogs/resizerecipedialog.h new file mode 100644 index 0000000..725ecdb --- /dev/null +++ b/src/dialogs/resizerecipedialog.h @@ -0,0 +1,69 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef RESIZERECIPEDIALOG_H +#define RESIZERECIPEDIALOG_H + +#include <kdialogbase.h> + +class Recipe; + +class TQVBoxLayout; +class TQHBoxLayout; +class TQGridLayout; +class FractionInput; +class TQButtonGroup; +class TQFrame; +class TQLabel; +class KIntNumInput; +class KLineEdit; +class TQRadioButton; +class TQPushButton; + +/** + *@author Jason Kivlighn + */ +class ResizeRecipeDialog : public KDialogBase +{ + TQ_OBJECT + +public: + ResizeRecipeDialog( TQWidget *parent, Recipe* ); + +protected slots: + void accept(); + void activateCurrentOption( int ); + virtual void languageChange(); + +private: + void resizeRecipe( double factor ); + + Recipe *m_recipe; + + TQButtonGroup* buttonGroup; + TQRadioButton* yieldRadioButton; + TQFrame* yieldFrame; + TQLabel* currentYieldLabel; + TQLabel* newYieldLabel; + KLineEdit* currentYieldInput; + FractionInput* newYieldInput; + TQRadioButton* factorRadioButton; + TQFrame* factorFrame; + TQLabel* factorLabel; + FractionInput* factorInput; + + TQVBoxLayout* buttonGroupLayout; + TQGridLayout* yieldFrameLayout; + TQHBoxLayout* factorFrameLayout; +}; + +#endif //RESIZERECIPEDIALOG_H diff --git a/src/dialogs/selectauthorsdialog.cpp b/src/dialogs/selectauthorsdialog.cpp new file mode 100644 index 0000000..d7583d3 --- /dev/null +++ b/src/dialogs/selectauthorsdialog.cpp @@ -0,0 +1,181 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "selectauthorsdialog.h" + +#include <tqmessagebox.h> +#include <tqvbox.h> + +#include <tdeconfig.h> +#include <kdialog.h> +#include <tdelocale.h> +#include <tdemessagebox.h> +#include <tdeglobal.h> + +#include "backends/recipedb.h" + +SelectAuthorsDialog::SelectAuthorsDialog( TQWidget *parent, const ElementList ¤tAuthors, RecipeDB *db ) + : KDialogBase( parent, "SelectAuthorsDialog", true, i18n("Authors"), + KDialogBase::Ok | KDialogBase::Cancel, KDialogBase::Ok ), + database(db) +{ + TQVBox *page = makeVBoxMainWidget(); + + //Design UI + + // Combo to Pick authors + TQHBox *topBox = new TQHBox(page); + topBox->setSpacing(6); + + authorsCombo = new KComboBox( true, topBox ); + authorsCombo->setSizePolicy( TQSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::MinimumExpanding ) ); + authorsCombo->completionObject() ->setCompletionMode( TDEGlobalSettings::CompletionPopupAuto ); + authorsCombo->lineEdit() ->disconnect( authorsCombo ); //so hitting enter doesn't enter the item into the box + + // Add/Remove buttons + + il = new TDEIconLoader; + addAuthorButton = new TQPushButton( topBox ); + TQPixmap pm = il->loadIcon( "go-down", TDEIcon::NoGroup, 16 ); + addAuthorButton->setIconSet( pm ); + + removeAuthorButton = new TQPushButton( topBox ); + pm = il->loadIcon( "go-up", TDEIcon::NoGroup, 16 ); + removeAuthorButton->setIconSet( pm ); + + // Author List + + authorListView = new TDEListView( page ); + authorListView->setSizePolicy( TQSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::MinimumExpanding ) ); + + TDEConfig * config = TDEGlobal::config(); + config->setGroup( "Advanced" ); + bool show_id = config->readBoolEntry( "ShowID", false ); + authorListView->addColumn( i18n( "Id" ), show_id ? -1 : 0 ); + authorListView->addColumn( i18n( "Author" ) ); + authorListView->setAllColumnsShowFocus( true ); + + // Load the list + loadAuthors( currentAuthors ); + + adjustSize(); + resize(450, height()); + + // Connect signals & Slots + connect ( addAuthorButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( addAuthor() ) ); + connect ( removeAuthorButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( removeAuthor() ) ); + + authorsCombo->setEditText(TQString::null); + authorsCombo->lineEdit()->setFocus(); +} + +SelectAuthorsDialog::~SelectAuthorsDialog() +{} + +void SelectAuthorsDialog::getSelectedAuthors( ElementList *newAuthors ) +{ + + for ( TQListViewItem * it = authorListView->firstChild();it; it = it->nextSibling() ) { + Element author; + author.id = it->text( 0 ).toInt(); + author.name = it->text( 1 ); + newAuthors->append( author ); + } + +} + +void SelectAuthorsDialog::loadAuthors( const ElementList ¤tAuthors ) +{ + + // Load the combo + reloadAuthorsCombo(); + + // Load the ListView with the authors of this recipe + authorListView->clear(); + for ( ElementList::const_iterator author_it = currentAuthors.begin(); author_it != currentAuthors.end(); ++author_it ) { + ( void ) new TQListViewItem( authorListView, TQString::number( ( *author_it ).id ), ( *author_it ).name ); + } + +} + +void SelectAuthorsDialog::addAuthor( void ) +{ + //check bounds first + if ( authorsCombo->currentText().length() > uint(database->maxAuthorNameLength()) ) { + KMessageBox::error( this, TQString( i18n( "Author name cannot be longer than %1 characters." ) ).arg( database->maxAuthorNameLength() ) ); + authorsCombo->lineEdit() ->selectAll(); + return ; + } + + if ( authorsCombo->lineEdit()->text().isEmpty() ) + return; + + if ( authorsCombo->contains( authorsCombo->currentText() ) ) + authorsCombo->setCurrentItem( authorsCombo->currentText() ); + + createNewAuthorIfNecessary(); + + int currentItem = authorsCombo->currentItem(); + Element currentElement = authorList.getElement( currentItem ); + + ( void ) new TQListViewItem( authorListView, TQString::number( currentElement.id ), currentElement.name ); + +} + +void SelectAuthorsDialog::removeAuthor( void ) +{ + // Find the selected item first + TQListViewItem * it; + it = authorListView->selectedItem(); + + if ( it ) { // Check if an author is selected first + delete it; + } + +} + +void SelectAuthorsDialog::createNewAuthorIfNecessary( void ) +{ + + if ( !authorsCombo->contains( authorsCombo->currentText() ) && + !( authorsCombo->currentText().stripWhiteSpace() ).isEmpty() ) // author is not in the list and is not empty + { // Create new author + TQString newAuthorName = authorsCombo->currentText(); + database->createNewAuthor( newAuthorName ); + //List again the authors + reloadAuthorsCombo(); + + // Select the newly created author + authorsCombo->setCurrentItem( newAuthorName ); + } +} + + +void SelectAuthorsDialog::reloadAuthorsCombo( void ) +{ + + //Load the author list + database->loadAuthors( &authorList ); + + // Load combo with all the authors + authorsCombo->clear(); + authorsCombo->completionObject() ->clear(); + + for ( ElementList::const_iterator author_it = authorList.begin(); author_it != authorList.end(); ++author_it ) { + authorsCombo->insertItem( ( *author_it ).name ); + authorsCombo->completionObject() ->addItem( ( *author_it ).name ); + } + +} + + +#include "selectauthorsdialog.moc" diff --git a/src/dialogs/selectauthorsdialog.h b/src/dialogs/selectauthorsdialog.h new file mode 100644 index 0000000..6304e08 --- /dev/null +++ b/src/dialogs/selectauthorsdialog.h @@ -0,0 +1,66 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef SELECTAUTHORSDIALOG_H +#define SELECTAUTHORSDIALOG_H + +#include <tqlayout.h> +#include <tqpushbutton.h> + +#include <kdialogbase.h> +#include <kiconloader.h> +#include <tdelistview.h> +#include <kcombobox.h> + +#include "datablocks/elementlist.h" + +class RecipeDB; + +/** +@author Unai Garro +*/ +class SelectAuthorsDialog: public KDialogBase +{ + + TQ_OBJECT + +public: + + SelectAuthorsDialog( TQWidget *parent, const ElementList ¤tAuthors, RecipeDB *db ); + ~SelectAuthorsDialog(); + void getSelectedAuthors( ElementList *newAuthors ); + +private: + + //Widgets + KComboBox *authorsCombo; + TDEListView *authorListView; + TQPushButton *okButton; + TQPushButton *cancelButton; + TQPushButton *addAuthorButton; + TQPushButton *removeAuthorButton; + TDEIconLoader *il; + + //Variables + ElementList authorList; + RecipeDB *database; + + //Private methods + void loadAuthors( const ElementList &authorList ); + void createNewAuthorIfNecessary( void ); + void reloadAuthorsCombo( void ); +private slots: + void addAuthor( void ); + void removeAuthor( void ); +}; + +#endif diff --git a/src/dialogs/selectcategoriesdialog.cpp b/src/dialogs/selectcategoriesdialog.cpp new file mode 100644 index 0000000..fe75bd3 --- /dev/null +++ b/src/dialogs/selectcategoriesdialog.cpp @@ -0,0 +1,104 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "selectcategoriesdialog.h" +#include "createcategorydialog.h" + +#include <tqvbox.h> + +#include <tdelocale.h> +#include <kdebug.h> +#include <kdialog.h> +#include <tdemessagebox.h> + +#include "datablocks/categorytree.h" +#include "backends/recipedb.h" +#include "widgets/categorylistview.h" + +SelectCategoriesDialog::SelectCategoriesDialog( TQWidget *parent, const ElementList &items_on, RecipeDB *db ) + : KDialogBase( parent, "SelectCategoriesDialog", true, i18n("Categories"), + KDialogBase::Ok | KDialogBase::Cancel, KDialogBase::Ok ), + database(db) +{ + TQVBox *page = makeVBoxMainWidget(); + + //Design UI + + //Category List + categoryListView = new CategoryCheckListView( page, db, true, items_on ); + categoryListView->reload(); + + //New category button + TQPushButton *newCatButton = new TQPushButton( page ); + newCatButton->setText( i18n( "&New Category..." ) ); + newCatButton->setFlat( true ); + + // Load the list + loadCategories( items_on ); + + setSizeGripEnabled( true ); + + // Connect signals & Slots + connect ( newCatButton, TQ_SIGNAL( clicked() ), TQ_SLOT( createNewCategory() ) ); +} + +SelectCategoriesDialog::~SelectCategoriesDialog() +{} + +void SelectCategoriesDialog::getSelectedCategories( ElementList *newSelected ) +{ + *newSelected = categoryListView->selections(); +} + +void SelectCategoriesDialog::loadCategories( const ElementList &items_on ) +{ + categoryListView->populateAll(); + + ElementList::const_iterator it; + for ( it = items_on.begin(); it != items_on.end(); ++it ) { + CategoryCheckListItem *new_item = (CategoryCheckListItem*)categoryListView->findItem(TQString::number((*it).id),1); + if ( new_item ) { + new_item->setOn(true); + } + } +} + +void SelectCategoriesDialog::createNewCategory( void ) +{ + ElementList categories; + database->loadCategories( &categories ); + CreateCategoryDialog* categoryDialog = new CreateCategoryDialog( this, categories ); + + if ( categoryDialog->exec() == TQDialog::Accepted ) { + TQString result = categoryDialog->newCategoryName(); + int subcategory = categoryDialog->subcategory(); + + //check bounds first + if ( result.length() > uint(database->maxCategoryNameLength()) ) { + KMessageBox::error( this, TQString( i18n( "Category name cannot be longer than %1 characters." ) ).arg( database->maxCategoryNameLength() ) ); + return ; + } + + database->createNewCategory( result, subcategory ); // Create the new category in the database + + //a listview item will automatically be created, but we need to turn it on + Element new_cat( result, database->lastInsertID() ); + TQCheckListItem *new_item = ((TQCheckListItem*)categoryListView->findItem( TQString::number(new_cat.id), 1 )); + if ( new_item ) + new_item->setOn(true); + } + + delete categoryDialog; +} + + +#include "selectcategoriesdialog.moc" diff --git a/src/dialogs/selectcategoriesdialog.h b/src/dialogs/selectcategoriesdialog.h new file mode 100644 index 0000000..d73bb2b --- /dev/null +++ b/src/dialogs/selectcategoriesdialog.h @@ -0,0 +1,57 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef SELECTCATEGORIESDIALOG_H +#define SELECTCATEGORIESDIALOG_H + +#include <tqlayout.h> +#include <tqpushbutton.h> + +#include <tdelistview.h> +#include <kdialogbase.h> + +#include "datablocks/elementlist.h" + +class CategoryTree; +class CategoryCheckListItem; +class CategoryCheckListView; +class RecipeDB; + +/** +@author Unai Garro +*/ +class SelectCategoriesDialog: public KDialogBase +{ + + TQ_OBJECT + +public: + + SelectCategoriesDialog( TQWidget *parent, const ElementList &items_on, RecipeDB* db ); + ~SelectCategoriesDialog(); + void getSelectedCategories( ElementList *selected ); +private: + + //Widgets + CategoryCheckListView *categoryListView; + + //Variables + RecipeDB *database; + + //Private methods + void loadCategories( const ElementList &items_on ); + +private slots: + void createNewCategory( void ); +}; + +#endif diff --git a/src/dialogs/selectpropertydialog.cpp b/src/dialogs/selectpropertydialog.cpp new file mode 100644 index 0000000..1ce1887 --- /dev/null +++ b/src/dialogs/selectpropertydialog.cpp @@ -0,0 +1,123 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "selectpropertydialog.h" + +#include <tdeconfig.h> +#include <tdeglobal.h> +#include <tdelocale.h> + +#include "datablocks/ingredientpropertylist.h" + +SelectPropertyDialog::SelectPropertyDialog( TQWidget* parent, IngredientPropertyList *propertyList, UnitList *unitList, OptionFlag showEmpty ) + : KDialogBase( parent, "SelectPropertyDialog", true, i18n( "Choose Property" ), + KDialogBase::Ok | KDialogBase::Cancel, KDialogBase::Ok ), m_showEmpty(showEmpty) +{ + + // Initialize internal variables + unitListBack = new UnitList; + + // Initialize Widgets + TQVBox *page = makeVBoxMainWidget(); + + box = new TQGroupBox( page ); + box->setTitle( i18n( "Choose Property" ) ); + box->setColumnLayout( 0, TQt::Vertical ); + box->layout() ->setSpacing( 6 ); + box->layout() ->setMargin( 11 ); + TQVBoxLayout *boxLayout = new TQVBoxLayout( box->layout() ); + boxLayout->setAlignment( TQt::AlignTop ); + + propertyChooseView = new TDEListView( box, "propertyChooseView" ); + + TDEConfig *config = TDEGlobal::config(); + config->setGroup( "Advanced" ); + bool show_id = config->readBoolEntry( "ShowID", false ); + propertyChooseView->addColumn( i18n( "Id" ), show_id ? -1 : 0 ); + + propertyChooseView->addColumn( i18n( "Property" ) ); + propertyChooseView->setAllColumnsShowFocus( true ); + boxLayout->addWidget( propertyChooseView ); + + TQHBoxLayout *layout2 = new TQHBoxLayout; + + perUnitsLabel = new TQLabel( box ); + perUnitsLabel->setGeometry( TQRect( 5, 285, 100, 30 ) ); + perUnitsLabel->setText( i18n( "Per units:" ) ); + layout2->addWidget( perUnitsLabel ); + + perUnitsBox = new KComboBox( FALSE, box ); + layout2->addWidget( perUnitsBox ); + boxLayout->addLayout( layout2 ); + + resize( TQSize( 200, 380 ).expandedTo( minimumSizeHint() ) ); + clearWState( WState_Polished ); + + // Load data + loadProperties( propertyList ); + loadUnits( unitList ); +} + + +SelectPropertyDialog::~SelectPropertyDialog() +{} + + +int SelectPropertyDialog::propertyID( void ) +{ + + TQListViewItem * it; + if ( ( it = propertyChooseView->selectedItem() ) ) { + return ( it->text( 0 ).toInt() ); + } + else + return ( -1 ); +} + +int SelectPropertyDialog::perUnitsID() +{ + + int comboCount = perUnitsBox->count(); + if ( comboCount > 0 ) { // If not, the list may be empty (no list defined) and crashes while reading as seen before. So check just in case. + int comboID = perUnitsBox->currentItem(); + return ( *unitListBack->at( comboID ) ).id; + } + else + return ( -1 ); +} + +void SelectPropertyDialog::loadProperties( IngredientPropertyList *propertyList ) +{ + for ( IngredientPropertyList::const_iterator prop_it = propertyList->begin(); prop_it != propertyList->end(); ++prop_it ) { + ( void ) new TQListViewItem( propertyChooseView, TQString::number( (*prop_it).id ), (*prop_it).name ); + } +} +void SelectPropertyDialog::loadUnits( UnitList *unitList ) +{ + for ( UnitList::const_iterator unit_it = unitList->begin(); unit_it != unitList->end(); ++unit_it ) { + TQString unitName = ( *unit_it ).name; + if ( unitName.isEmpty() ) { + if ( m_showEmpty == ShowEmptyUnit ) + unitName = " "+i18n("-No unit-"); + else + continue; + } + + // Insert in the combobox + perUnitsBox->insertItem( unitName ); + + // Store with index for using later + Unit newUnit( *unit_it ); + newUnit.name = unitName; + unitListBack->append( newUnit ); + } +} diff --git a/src/dialogs/selectpropertydialog.h b/src/dialogs/selectpropertydialog.h new file mode 100644 index 0000000..94d7c3b --- /dev/null +++ b/src/dialogs/selectpropertydialog.h @@ -0,0 +1,60 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef SELECTPROPERTYDIALOG_H +#define SELECTPROPERTYDIALOG_H + +#include <tqwidget.h> +#include <tqlayout.h> +#include <tqgroupbox.h> +#include <tqlabel.h> +#include <tqpushbutton.h> +#include <tqvbox.h> + +#include <tdelistview.h> +#include <kcombobox.h> +#include <kdialogbase.h> + +#include "datablocks/unit.h" + +class IngredientPropertyList; + +/** +@author Unai Garro +*/ +class SelectPropertyDialog: public KDialogBase +{ +public: + enum OptionFlag { ShowEmptyUnit, HideEmptyUnit } ; + + // Methods + SelectPropertyDialog( TQWidget* parent, IngredientPropertyList *propertyList, UnitList *unitList, OptionFlag showEmpty = ShowEmptyUnit ); + ~SelectPropertyDialog(); + int propertyID( void ); + int perUnitsID( void ); +private: + //Widgets + TQGroupBox *box; + TDEListView *propertyChooseView; + TQLabel *perUnitsLabel; + KComboBox *perUnitsBox; + OptionFlag m_showEmpty; + + void loadProperties( IngredientPropertyList *propertyList ); + void loadUnits( UnitList *unitList ); + + //Internal variables + UnitList *unitListBack; // To store unit list with ID's for later use + +}; + +#endif diff --git a/src/dialogs/selectrecipedialog.cpp b/src/dialogs/selectrecipedialog.cpp new file mode 100644 index 0000000..a8a6ca6 --- /dev/null +++ b/src/dialogs/selectrecipedialog.cpp @@ -0,0 +1,257 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "selectrecipedialog.h" + +#include <tqsignalmapper.h> +#include <tqtabwidget.h> +#include <tqtooltip.h> + +#include <tdelocale.h> +#include <kdebug.h> +#include <tdeapplication.h> +#include <kprogress.h> +#include <tdemessagebox.h> +#include <tdeglobal.h> +#include <tdeconfig.h> +#include <kcursor.h> +#include <kiconloader.h> + +#include "advancedsearchdialog.h" +#include "datablocks/categorytree.h" +#include "backends/recipedb.h" +#include "datablocks/recipe.h" +#include "selectunitdialog.h" +#include "createelementdialog.h" +#include "recipefilter.h" +#include "widgets/recipelistview.h" +#include "widgets/categorylistview.h" +#include "widgets/categorycombobox.h" + +SelectRecipeDialog::SelectRecipeDialog( TQWidget *parent, RecipeDB* db ) + : TQWidget( parent ) +{ + //Store pointer to Recipe Database + database = db; + + TQVBoxLayout *tabLayout = new TQVBoxLayout( this ); + tabWidget = new TQTabWidget( this ); + tabWidget->setSizePolicy( TQSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::MinimumExpanding ) ); + tabLayout->addWidget( tabWidget ); + + basicSearchTab = new TQGroupBox( this ); + basicSearchTab->setFrameStyle( TQFrame::NoFrame ); + basicSearchTab->setSizePolicy( TQSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::MinimumExpanding ) ); + + //Design dialog + + layout = new TQGridLayout( basicSearchTab, 1, 1, 0, 0 ); + + // Border Spacers + TQSpacerItem* spacer_left = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::Minimum ); + layout->addMultiCell( spacer_left, 1, 4, 0, 0 ); + TQSpacerItem* spacer_top = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addMultiCell( spacer_top, 0, 0, 1, 4 ); + + searchBar = new TQHBox( basicSearchTab ); + searchBar->setSpacing( 7 ); + layout->addWidget( searchBar, 1, 1 ); + + TDEIconLoader *il = new TDEIconLoader; + TQPushButton *clearSearchButton = new TQPushButton( searchBar ); + clearSearchButton->setPixmap( il->loadIcon( "locationbar_erase", TDEIcon::NoGroup, 16 ) ); + + searchLabel = new TQLabel( searchBar ); + searchLabel->setText( i18n( "Search:" ) ); + searchLabel->setFixedWidth( searchLabel->fontMetrics().width( i18n( "Search:" ) ) + 5 ); + searchBox = new KLineEdit( searchBar ); + + TQSpacerItem* searchSpacer = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::Minimum ); + layout->addItem( searchSpacer, 1, 2 ); + + #ifdef ENABLE_SLOW + categoryBox = new CategoryComboBox( basicSearchTab, database ); + layout->addWidget( categoryBox, 1, 3 ); + #endif + + TQSpacerItem* spacerFromSearchBar = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( spacerFromSearchBar, 2, 1 ); + + recipeListView = new RecipeListView( basicSearchTab, database ); + recipeListView->reload(); + recipeListView->setSizePolicy( TQSizePolicy::Ignored, TQSizePolicy::Expanding ); + layout->addMultiCellWidget( recipeListView, 3, 3, 1, 3 ); + + buttonBar = new TQHBox( basicSearchTab ); + layout->addMultiCellWidget( buttonBar, 4, 4, 1, 3 ); + + openButton = new TQPushButton( buttonBar ); + openButton->setText( i18n( "Open Recipe(s)" ) ); + openButton->setDisabled( true ); + TQPixmap pm = il->loadIcon( "ok", TDEIcon::NoGroup, 16 ); + openButton->setIconSet( pm ); + editButton = new TQPushButton( buttonBar ); + editButton->setText( i18n( "Edit Recipe" ) ); + editButton->setDisabled( true ); + pm = il->loadIcon( "edit", TDEIcon::NoGroup, 16 ); + editButton->setIconSet( pm ); + removeButton = new TQPushButton( buttonBar ); + removeButton->setText( i18n( "Delete" ) ); + removeButton->setDisabled( true ); + removeButton->setMaximumWidth( 100 ); + pm = il->loadIcon( "editshred", TDEIcon::NoGroup, 16 ); + removeButton->setIconSet( pm ); + + tabWidget->insertTab( basicSearchTab, i18n( "Basic" ) ); + + advancedSearch = new AdvancedSearchDialog( this, database ); + tabWidget->insertTab( advancedSearch, i18n( "Advanced" ) ); + + TQToolTip::add( clearSearchButton, i18n( "Clear search" ) ); + + //Takes care of all recipe actions and provides a popup menu to 'recipeListView' + actionHandler = new RecipeActionsHandler( recipeListView, database ); + + recipeFilter = new RecipeFilter( recipeListView ); + + // Signals & Slots + + connect( openButton, TQ_SIGNAL( clicked() ), actionHandler, TQ_SLOT( open() ) ); + connect( this, TQ_SIGNAL( recipeSelected( bool ) ), openButton, TQ_SLOT( setEnabled( bool ) ) ); + connect( editButton, TQ_SIGNAL( clicked() ), actionHandler, TQ_SLOT( edit() ) ); + connect( this, TQ_SIGNAL( recipeSelected( bool ) ), editButton, TQ_SLOT( setEnabled( bool ) ) ); + connect( removeButton, TQ_SIGNAL( clicked() ), actionHandler, TQ_SLOT( remove() ) ); + connect( this, TQ_SIGNAL( recipeSelected( bool ) ), removeButton, TQ_SLOT( setEnabled( bool ) ) ); + + connect( clearSearchButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( clearSearch() ) ); + + TDEConfig * config = kapp->config(); + config->setGroup( "Performance" ); + if ( config->readBoolEntry("SearchAsYouType",true) ) { + connect( searchBox, TQ_SIGNAL( returnPressed( const TQString& ) ), recipeFilter, TQ_SLOT( filter( const TQString& ) ) ); + connect( searchBox, TQ_SIGNAL( textChanged( const TQString& ) ), this, TQ_SLOT( ensurePopulated() ) ); + connect( searchBox, TQ_SIGNAL( textChanged( const TQString& ) ), recipeFilter, TQ_SLOT( filter( const TQString& ) ) ); + } + else { + connect( searchBox, TQ_SIGNAL( returnPressed( const TQString& ) ), this, TQ_SLOT( ensurePopulated() ) ); + connect( searchBox, TQ_SIGNAL( returnPressed( const TQString& ) ), recipeFilter, TQ_SLOT( filter( const TQString& ) ) ); + } + + connect( recipeListView, TQ_SIGNAL( selectionChanged() ), this, TQ_SLOT( haveSelectedItems() ) ); + #ifdef ENABLE_SLOW + connect( recipeListView, TQ_SIGNAL( nextGroupLoaded() ), categoryBox, TQ_SLOT( loadNextGroup() ) ); + connect( recipeListView, TQ_SIGNAL( prevGroupLoaded() ), categoryBox, TQ_SLOT( loadPrevGroup() ) ); + connect( categoryBox, TQ_SIGNAL( activated( int ) ), this, TQ_SLOT( filterComboCategory( int ) ) ); + #endif + connect( recipeListView, TQ_SIGNAL( nextGroupLoaded() ), TQ_SLOT( refilter() ) ); + connect( recipeListView, TQ_SIGNAL( prevGroupLoaded() ), TQ_SLOT( refilter() ) ); + + connect( advancedSearch, TQ_SIGNAL( recipeSelected( int, int ) ), TQ_SIGNAL( recipeSelected( int, int ) ) ); + connect( advancedSearch, TQ_SIGNAL( recipesSelected( const TQValueList<int> &, int ) ), TQ_SIGNAL( recipesSelected( const TQValueList<int> &, int ) ) ); + + connect( actionHandler, TQ_SIGNAL( recipeSelected( int, int ) ), TQ_SIGNAL( recipeSelected( int, int ) ) ); + connect( actionHandler, TQ_SIGNAL( recipesSelected( const TQValueList<int> &, int ) ), TQ_SIGNAL( recipesSelected( const TQValueList<int> &, int ) ) ); + + delete il; +} + +SelectRecipeDialog::~SelectRecipeDialog() +{ + delete recipeFilter; +} + +void SelectRecipeDialog::clearSearch() +{ + searchBox->setText( TQString::null ); + recipeFilter->filter( TQString::null ); +} + +void SelectRecipeDialog::reload( ReloadFlags flag ) +{ + recipeListView->reload(flag); + + #ifdef ENABLE_SLOW + categoryBox->reload(); + filterComboCategory( categoryBox->currentItem() ); + #endif +} + +void SelectRecipeDialog::refilter() +{ + if ( !searchBox->text().isEmpty() ) { + ensurePopulated(); + recipeFilter->filter(searchBox->text()); + } +} + +void SelectRecipeDialog::ensurePopulated() +{ + recipeListView->populateAll(); +} + +void SelectRecipeDialog::haveSelectedItems() +{ + if ( recipeListView->selectedItems().count() > 0 ) + emit recipeSelected( true ); + else + emit recipeSelected( false ); +} + +void SelectRecipeDialog::getCurrentRecipe( Recipe *recipe ) +{ + TQPtrList<TQListViewItem> items = recipeListView->selectedItems(); + if ( items.count() == 1 && items.at(0)->rtti() == 1000 ) { + RecipeListItem * recipe_it = ( RecipeListItem* )items.at(0); + database->loadRecipe( recipe, RecipeDB::All, recipe_it->recipeID() ); + } +} + +void SelectRecipeDialog::filterComboCategory( int row ) +{ + recipeListView->populateAll(); //TODO: this would be faster if we didn't need to load everything first + + kdDebug() << "I got row " << row << "\n"; + + //First get the category ID corresponding to this combo row + int categoryID = categoryBox->id( row ); + + //Now filter + recipeFilter->filterCategory( categoryID ); // if categoryID==-1 doesn't filter + recipeFilter->filter( searchBox->text() ); + + if ( categoryID != -1 ) { + TQListViewItemIterator it( recipeListView ); + while ( it.current() ) { + TQListViewItem *item = it.current(); + if ( item->isVisible() ) { + item->setOpen( true ); //will only open if already populated + //(could be the selected category's parent + if ( !item->firstChild() ) { + recipeListView->open( item ); //populates and opens the selected category + break; + } + } + ++it; + } + + } +} + +RecipeActionsHandler* SelectRecipeDialog::getActionsHandler() const +{ + if ( tabWidget->currentPage() == basicSearchTab ) + return actionHandler; + else + return advancedSearch->actionHandler; +} + +#include "selectrecipedialog.moc" diff --git a/src/dialogs/selectrecipedialog.h b/src/dialogs/selectrecipedialog.h new file mode 100644 index 0000000..d296259 --- /dev/null +++ b/src/dialogs/selectrecipedialog.h @@ -0,0 +1,99 @@ +/************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef SELECTRECIPEDIALOG_H +#define SELECTRECIPEDIALOG_H + + +#include <tqwidget.h> +#include <tqpushbutton.h> +#include <tqlabel.h> +#include <tqlayout.h> +#include <tqcursor.h> +#include <tqintdict.h> + +#include <kcombobox.h> +#include <klineedit.h> +#include <tdelistview.h> +#include <kdialog.h> +#include <tdefiledialog.h> +#include <tdepopupmenu.h> + +#include "recipeactionshandler.h" +#include "widgets/dblistviewbase.h" + +class TQHBox; +class TQGroupBox; + +class RecipeDB; +class ElementList; +class Recipe; +class AdvancedSearchDialog; +class CategoryTree; +class RecipeFilter; +class RecipeListView; +class CategoryComboBox; + +/** +@author Unai Garro +*/ +class SelectRecipeDialog : public TQWidget +{ + TQ_OBJECT +public: + SelectRecipeDialog( TQWidget *parent, RecipeDB *db ); + ~SelectRecipeDialog(); + + + //Public Methods + void getCurrentRecipe( Recipe *recipe ); + + RecipeActionsHandler * getActionsHandler() const; + +private: + + // Widgets + TQGridLayout *layout; + TQTabWidget *tabWidget; + TQGroupBox *basicSearchTab; + TQHBox *searchBar; + RecipeListView* recipeListView; + TQHBox *buttonBar; + TQPushButton *openButton; + TQPushButton *removeButton; + TQPushButton *editButton; + TQLabel *searchLabel; + KLineEdit *searchBox; + CategoryComboBox *categoryBox; + AdvancedSearchDialog *advancedSearch; + // Internal Data + RecipeDB *database; + RecipeActionsHandler *actionHandler; + RecipeFilter *recipeFilter; + +signals: + void recipeSelected( int id, int action ); + void recipesSelected( const TQValueList<int> &ids, int action ); + void recipeSelected( bool ); + +private slots: + void filterComboCategory( int row ); + void refilter(); + void ensurePopulated(); + +public slots: + void haveSelectedItems(); + void reload( ReloadFlags flag = Load ); + void clearSearch(); +}; + +#endif diff --git a/src/dialogs/selectunitdialog.cpp b/src/dialogs/selectunitdialog.cpp new file mode 100644 index 0000000..d9500a6 --- /dev/null +++ b/src/dialogs/selectunitdialog.cpp @@ -0,0 +1,77 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "selectunitdialog.h" + +#include <tdeconfig.h> +#include <tdeglobal.h> +#include <tdelocale.h> + +SelectUnitDialog::SelectUnitDialog( TQWidget* parent, const UnitList &unitList, OptionFlag showEmpty ) + : KDialogBase( parent, "SelectUnitDialog", true, i18n( "Choose Unit" ), + KDialogBase::Ok | KDialogBase::Cancel, KDialogBase::Ok ), m_showEmpty(showEmpty) +{ + TQVBox *page = makeVBoxMainWidget(); + + box = new TQGroupBox( page ); + box->setTitle( i18n( "Choose Unit" ) ); + box->setColumnLayout( 0, TQt::Vertical ); + TQVBoxLayout *boxLayout = new TQVBoxLayout( box->layout() ); + + unitChooseView = new TDEListView( box ); + + TDEConfig *config = TDEGlobal::config(); + config->setGroup( "Advanced" ); + bool show_id = config->readBoolEntry( "ShowID", false ); + unitChooseView->addColumn( i18n( "Id" ), show_id ? -1 : 0 ); + + unitChooseView->addColumn( i18n( "Unit" ) ); + unitChooseView->setSorting(1); + unitChooseView->setGeometry( TQRect( 5, 30, 180, 250 ) ); + unitChooseView->setAllColumnsShowFocus( true ); + boxLayout->addWidget( unitChooseView ); + + resize( TQSize( 200, 350 ) ); + + loadUnits( unitList ); +} + + +SelectUnitDialog::~SelectUnitDialog() +{} + +int SelectUnitDialog::unitID( void ) +{ + + TQListViewItem * it; + if ( ( it = unitChooseView->selectedItem() ) ) { + return ( it->text( 0 ).toInt() ); + } + else + return ( -1 ); +} + +void SelectUnitDialog::loadUnits( const UnitList &unitList ) +{ + for ( UnitList::const_iterator unit_it = unitList.begin(); unit_it != unitList.end(); ++unit_it ) { + TQString unitName = ( *unit_it ).name; + if ( unitName.isEmpty() ) { + if ( m_showEmpty == ShowEmptyUnit ) + unitName = " "+i18n("-No unit-"); + else + continue; + } + + ( void ) new TQListViewItem( unitChooseView, TQString::number( ( *unit_it ).id ), unitName ); + } +} + diff --git a/src/dialogs/selectunitdialog.h b/src/dialogs/selectunitdialog.h new file mode 100644 index 0000000..f2d8af0 --- /dev/null +++ b/src/dialogs/selectunitdialog.h @@ -0,0 +1,51 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef SELECTUNITDIALOG_H +#define SELECTUNITDIALOG_H + +#include <tqwidget.h> +#include <tqlayout.h> +#include <tqpushbutton.h> +#include <tqgroupbox.h> +#include <tqvbox.h> + +#include <tdelistview.h> +#include <kdialogbase.h> + +#include "datablocks/unit.h" + +/** +@author Unai Garro +*/ +class SelectUnitDialog : public KDialogBase +{ +public: + enum OptionFlag { ShowEmptyUnit, HideEmptyUnit } ; + + SelectUnitDialog( TQWidget* parent, const UnitList &unitList, OptionFlag = ShowEmptyUnit ); + + ~SelectUnitDialog(); + + int unitID( void ); + +private: + //Widgets + TQGroupBox *box; + TDEListView *unitChooseView; + OptionFlag m_showEmpty; + + void loadUnits( const UnitList &unitList ); + +}; + +#endif diff --git a/src/dialogs/setupdisplay.cpp b/src/dialogs/setupdisplay.cpp new file mode 100644 index 0000000..6efa42f --- /dev/null +++ b/src/dialogs/setupdisplay.cpp @@ -0,0 +1,602 @@ +/*************************************************************************** +* Copyright (C) 2003-2005 * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "setupdisplay.h" + +#include <tdeapplication.h> +#include <tdeconfig.h> +#include <kdebug.h> +#include <tdefontdialog.h> +#include <kcolordialog.h> +#include <tdelocale.h> +#include <tdepopupmenu.h> +#include <kiconloader.h> +#include <kstandarddirs.h> +#include <tdetempfile.h> + +#include <tdehtmlview.h> +#include <dom/dom_doc.h> +#include <dom/css_rule.h> + +#include <tqinputdialog.h> +#include <tqaction.h> +#include <tqlabel.h> +#include <tqfile.h> +#include <tqregexp.h> +#include <tqtextedit.h> +#include <tqtooltip.h> +#include <tqobjectlist.h> +#include <tqvaluelist.h> +#include <tqlayout.h> + +#include "datablocks/mixednumber.h" +#include "dialogs/borderdialog.h" +#include "exporters/htmlexporter.h" + +#include <cmath> + +KreDisplayItem::KreDisplayItem( const TQString &n, const TQString &_name ) : nodeId(n), name(_name) +{ + clear(); +} + +void KreDisplayItem::clear() +{ + alignment = TQt::AlignHCenter; + show = true; + backgroundColor = TQColor(255,255,255); + textColor = TQColor(0,0,0); + columns = 1; +} + +SetupDisplay::SetupDisplay( const Recipe &sample, TQWidget *parent ) : TDEHTMLPart(parent), + box_properties( new PropertiesMap ), + node_item_map( new TQMap<TQString, KreDisplayItem*> ), + has_changes( false ), + popup(0) +{ + connect( this, TQ_SIGNAL( popupMenu(const TQString &,const TQPoint &) ), TQ_SLOT( nodeClicked(const TQString &,const TQPoint &) ) ); + + if ( sample.recipeID != -1 ) + m_sample = sample; + else { + m_sample.title = i18n("Recipe Title"); + m_sample.yield.amount = 0; + m_sample.categoryList.append( Element(i18n( "Category 1, Category 2, ..." ) ) ); + m_sample.instructions = i18n("Instructions"); + m_sample.prepTime = TQTime(0,0); + + m_sample.authorList.append( Element(i18n( "Author 1, Author 2, ..." )) ); + + Ingredient ing; + ing.name = i18n("Ingredient 1"); + m_sample.ingList.append( ing ); + + ing.name = i18n("Ingredient 2"); + m_sample.ingList.append( ing ); + + ing.name = "..."; + m_sample.ingList.append( ing ); + + RatingCriteria rc; + Rating rating1; + rating1.rater = i18n("Rater"); + rating1.comment = i18n("Comment"); + + rc.name = i18n("Criteria 1"); + rc.stars = 5.0; + rating1.append(rc); + + rc.name = i18n("Criteria 2"); + rc.stars = 2.5; + rating1.append(rc); + + IngredientProperty prop; + prop.name = i18n("Property 1"); + m_sample.properties.append(prop); + prop.name = i18n("Property 2"); + m_sample.properties.append(prop); + prop.name = "..."; + m_sample.properties.append(prop); + + m_sample.ratingList.append(rating1); + } + + kdDebug()<<"first load"<<endl; + loadHTMLView(); + show(); + + createItem( "background", i18n("Background"), BackgroundColor ); + createItem( "title", i18n("Title"), Font | BackgroundColor | TextColor | Visibility | Alignment | Border ); + createItem( "instructions", i18n("Instructions"), Font | BackgroundColor | TextColor | Visibility | Alignment | Border ); + createItem( "prep_time", i18n("Preparation Time"), Font | BackgroundColor | TextColor | Visibility | Alignment | Border ); + createItem( "photo", i18n("Photo"), Visibility | Border ); + createItem( "authors", i18n("Authors"), Font | BackgroundColor | TextColor | Visibility | Alignment | Border ); + createItem( "categories", i18n("Categories"), Font | BackgroundColor | TextColor | Visibility | Alignment | Border ); + createItem( "ingredients", i18n("Ingredients"), Font | BackgroundColor | TextColor | Visibility | Alignment | Border | Columns ); + createItem( "properties", i18n("Properties"), Font | BackgroundColor | TextColor | Visibility | Alignment | Border | Columns ); + createItem( "ratings", i18n("Ratings"), Font | BackgroundColor | TextColor | Visibility | Alignment | Border ); + createItem( "yield", i18n("Yield"), Font | BackgroundColor | TextColor | Visibility | Alignment | Border ); +} + +SetupDisplay::~SetupDisplay() +{ + delete box_properties; + delete node_item_map; +} + +void SetupDisplay::loadHTMLView( const TQString &templateFile, const TQString &styleFile ) +{ + kdDebug()<<"loading template: "<<templateFile<<" style: "<<styleFile<<endl; + TQString tmp_filename = locateLocal( "tmp", "krecipes_recipe_view" ); + HTMLExporter exporter( tmp_filename + ".html", "html" ); + if ( templateFile != TQString::null ) + exporter.setTemplate( templateFile ); + if ( styleFile != TQString::null ) + exporter.setStyle( styleFile ); + + RecipeList recipeList; + recipeList.append(m_sample); + + TQFile file(tmp_filename + ".html"); + if ( file.open( IO_WriteOnly ) ) { + TQTextStream stream(&file); + exporter.writeStream(stream,recipeList); + } + else { + kdDebug()<<"Unable to open file for writing"<<endl; + } + file.close(); + + KURL url; + url.setPath( tmp_filename + ".html" ); + openURL( url ); + kdDebug() << "Opening URL: " << url.htmlURL() << endl; +} + +void SetupDisplay::reload() +{ + loadHTMLView( m_activeTemplate, m_activeStyle ); +} + +void SetupDisplay::loadTemplate( const TQString &filename ) +{ + bool storeChangedState = has_changes; + KTempFile tmpFile; + saveLayout(tmpFile.name()); + has_changes = storeChangedState; //saveLayout() sets changes to false + + loadHTMLView( filename, tmpFile.name() ); + + m_activeTemplate = filename; +} + +void SetupDisplay::createItem( const TQString &node, const TQString &name, unsigned int properties ) +{ + KreDisplayItem * item = new KreDisplayItem( node, name ); + box_properties->insert( item, properties ); + node_item_map->insert( node, item ); +} + +void SetupDisplay::loadLayout( const TQString &filename ) +{ + TQFile input( filename ); + if ( input.open( IO_ReadOnly ) ) { + TQDomDocument doc; + TQString error; + int line; + int column; + if ( !doc.setContent( &input, &error, &line, &column ) ) { + kdDebug() << TQString( i18n( "\"%1\" at line %2, column %3. This may not be a Krecipes layout file." ) ).arg( error ).arg( line ).arg( column ) << endl; + return ; + } + + m_styleSheet = DOM::CSSStyleSheet(); + + TQMap<TQString,KreDisplayItem*>::iterator it; + for ( it = node_item_map->begin(); it != node_item_map->end(); ++it ) { + it.data()->clear(); + } + processDocument( doc ); + + loadHTMLView(m_activeTemplate, filename); + m_activeStyle = filename; + + has_changes = false; + } + else + kdDebug() << "Unable to open file: " << filename << endl; +} + +void SetupDisplay::beginObject( const TQString &object ) +{ + TQMap<TQString, KreDisplayItem*>::iterator map_it = node_item_map->find( object ); + if ( map_it != node_item_map->end() ) + m_currentItem = map_it.data(); + else + m_currentItem = 0; +} + +void SetupDisplay::endObject() +{ + m_currentItem = 0; +} + +void SetupDisplay::loadBackgroundColor( const TQString &object, const TQColor &color ) +{ + if ( m_currentItem ) { + m_currentItem->backgroundColor = color; + m_styleSheet.insertRule("."+object+" { "+bgColorAsCSS(color)+" }",m_styleSheet.cssRules().length()); + } +} + +void SetupDisplay::loadFont( const TQString &object, const TQFont &font ) +{ + if ( m_currentItem ) { + m_currentItem->font = font; + m_styleSheet.insertRule("."+object+" { "+fontAsCSS(font)+" }",m_styleSheet.cssRules().length()); + } +} + +void SetupDisplay::loadTextColor( const TQString &object, const TQColor &color ) +{ + if ( m_currentItem ) { + m_currentItem->textColor = color; + m_styleSheet.insertRule("."+object+" { "+textColorAsCSS(color)+" }",m_styleSheet.cssRules().length()); + } +} + +void SetupDisplay::loadVisibility( const TQString &object, bool visible ) +{ + if ( m_currentItem ) { + m_currentItem->show = visible; + emit itemVisibilityChanged( m_currentItem, visible ); + + m_styleSheet.insertRule("."+object+" { "+visibilityAsCSS(visible)+" }",m_styleSheet.cssRules().length()); + } +} + +void SetupDisplay::loadAlignment( const TQString &object, int alignment ) +{ + if ( m_currentItem ) { + m_currentItem->alignment = alignment; + m_styleSheet.insertRule("."+object+" { "+alignmentAsCSS(alignment)+" }",m_styleSheet.cssRules().length()); + } +} + +void SetupDisplay::loadBorder( const TQString &object, const KreBorder& border ) +{ + if ( m_currentItem ) { + m_currentItem->border = border; + m_styleSheet.insertRule("."+object+" { "+borderAsCSS(border)+" }",m_styleSheet.cssRules().length()); + } +} + +void SetupDisplay::loadColumns( const TQString &/*object*/, int cols ) +{ + if ( m_currentItem ) { + m_currentItem->columns = cols; + } +} + +void SetupDisplay::saveLayout( const TQString &filename ) +{ + TQDomImplementation dom_imp; + TQDomDocument doc = dom_imp.createDocument( TQString::null, "krecipes-layout", dom_imp.createDocumentType( "krecipes-layout", TQString::null, TQString::null ) ); + + TQDomElement layout_tag = doc.documentElement(); + layout_tag.setAttribute( "version", 0.4 ); + //layout_tag.setAttribute( "generator", TQString("Krecipes v%1").arg(krecipes_version()) ); + doc.appendChild( layout_tag ); + + for ( TQMap<TQString, KreDisplayItem*>::const_iterator it = node_item_map->begin(); it != node_item_map->end(); ++it ) { + TQDomElement base_tag = doc.createElement( it.key() ); + layout_tag.appendChild( base_tag ); + + int properties = (*box_properties)[it.data()]; + if ( properties & BackgroundColor ) { + TQDomElement backgroundcolor_tag = doc.createElement( "background-color" ); + backgroundcolor_tag.appendChild( doc.createTextNode( it.data()->backgroundColor.name() ) ); + base_tag.appendChild( backgroundcolor_tag ); + } + + if ( properties & TextColor ) { + TQDomElement textcolor_tag = doc.createElement( "text-color" ); + textcolor_tag.appendChild( doc.createTextNode( it.data()->textColor.name() ) ); + base_tag.appendChild( textcolor_tag ); + } + + if ( properties & Font ) { + TQDomElement font_tag = doc.createElement( "font" ); + font_tag.appendChild( doc.createTextNode( it.data()->font.toString() ) ); + base_tag.appendChild( font_tag ); + } + + if ( properties & Visibility ) { + TQDomElement visibility_tag = doc.createElement( "visible" ); + visibility_tag.appendChild( doc.createTextNode( (it.data()->show) ? "true" : "false" ) ); + base_tag.appendChild( visibility_tag ); + } + + if ( properties & Alignment ) { + TQDomElement alignment_tag = doc.createElement( "alignment" ); + alignment_tag.appendChild( doc.createTextNode( TQString::number( it.data()->alignment ) ) ); + base_tag.appendChild( alignment_tag ); + } + + if ( properties & Border ) { + TQDomElement border_tag = doc.createElement( "border" ); + border_tag.setAttribute( "width", it.data()->border.width ); + border_tag.setAttribute( "style", it.data()->border.style ); + border_tag.setAttribute( "color", it.data()->border.color.name() ); + base_tag.appendChild( border_tag ); + } + + if ( properties & Columns ) { + TQDomElement columns_tag = doc.createElement( "columns" ); + columns_tag.appendChild( doc.createTextNode( TQString::number( it.data()->columns ) ) ); + base_tag.appendChild( columns_tag ); + } + } + + TQFile out_file( filename ); + if ( out_file.open( IO_WriteOnly ) ) { + has_changes = false; + + TQTextStream stream( &out_file ); + stream << "<?xml version=\"1.0\" encoding=\"UTF-8\" ?>\n" << doc.toString(); + } + else + kdDebug() << "Error: Unable to write to file " << filename << endl; +} + +void SetupDisplay::begin(const KURL &url, int xOffset, int yOffset) +{ + kdDebug()<<"begin"<<endl; + TDEHTMLPart::begin(url,xOffset,yOffset); + kdDebug()<<"end"<<endl; + + DOM::Document doc = document(); + DOM::DOMImplementation impl = doc.implementation(); + kdDebug() << "(1) document: " << document().handle() << endl; + #if 0 + if ( !impl.isNull() ) { + //m_styleSheet = impl.createCSSStyleSheet("-krecipes","screen"); + //m_styleSheet = DOM::CSSStyleSheet(); + //doc.addStyleSheet(m_styleSheet); + //applyStylesheet(); + } + #endif +} + +void SetupDisplay::nodeClicked(const TQString &/*url*/,const TQPoint &point) +{ + DOM::Node node = nodeUnderMouse(); + DOM::Element element; + if ( node.nodeType() != DOM::Node::ELEMENT_NODE ) { + kdDebug()<<"not an element"<<endl; + element = (DOM::Element)node.parentNode(); + } + else + element = (DOM::Element)node; + + while ( !element.parentNode().isNull() ) { + if ( element.hasAttribute("class") ) { + TQString id = element.getAttribute("class").string(); + if ( node_item_map->keys().contains(id) ) + break; + } + + element = (DOM::Element)element.parentNode(); + } + + m_currNodeId = element.getAttribute("class").string(); + if ( m_currNodeId.isEmpty() ) { + kdDebug()<<"Code error: unable to determine class of selected element"<<endl; + return; + } + + KreDisplayItem *item = *node_item_map->find( m_currNodeId ); + + delete popup; + popup = new TDEPopupMenu( view() ); + popup->insertTitle( item->name ); + + unsigned int properties = 0; + for ( PropertiesMap::const_iterator it = box_properties->begin(); it != box_properties->end(); ++it ) { + if ( it.key()->nodeId == m_currNodeId ) { + properties = it.data(); + break; + } + } + + TDEIconLoader il; + + if ( properties & BackgroundColor ) + popup->insertItem( i18n( "Background Color..." ), this, TQ_SLOT( setBackgroundColor() ) ); + + if ( properties & TextColor ) + popup->insertItem( i18n( "Text Color..." ), this, TQ_SLOT( setTextColor() ) ); + + if ( properties & Font ) + popup->insertItem( il.loadIconSet( "text", TDEIcon::Small, 16 ), i18n( "Font..." ), this, TQ_SLOT( setFont() ) ); + + if ( properties & Visibility ) { + int id = popup->insertItem( i18n( "Show" ), this, TQ_SLOT( setShown( int ) ) ); + popup->setItemChecked( id, item->show ); + } + + if ( properties & Alignment ) { + TQPopupMenu * sub_popup = new TQPopupMenu( popup ); + + TQActionGroup *alignment_actions = new TQActionGroup( this ); + alignment_actions->setExclusive( true ); + + TQAction *c_action = new TQAction( i18n( "Center" ), i18n( "Center" ), 0, alignment_actions, 0, true ); + TQAction *l_action = new TQAction( i18n( "Left" ), i18n( "Left" ), 0, alignment_actions, 0, true ); + TQAction *r_action = new TQAction( i18n( "Right" ), i18n( "Right" ), 0, alignment_actions, 0, true ); + + int align = item->alignment; + if ( align & TQt::AlignHCenter ) + c_action->setOn(true); + if ( align & TQt::AlignLeft ) + l_action->setOn(true); + if ( align & TQt::AlignRight ) + r_action->setOn(true); + + connect( alignment_actions, TQ_SIGNAL( selected( TQAction* ) ), TQ_SLOT( setAlignment( TQAction* ) ) ); + + popup->insertItem( i18n( "Alignment" ), sub_popup ); + + alignment_actions->addTo( sub_popup ); + } + + if ( properties & Border ) + popup->insertItem( i18n( "Border..." ), this, TQ_SLOT( setBorder() ) ); + + if ( properties & Columns ) + popup->insertItem( i18n( "Columns..." ), this, TQ_SLOT( setColumns() ) ); + + popup->popup( point ); +} + +void SetupDisplay::applyStylesheet() +{ + loadTemplate( m_activeTemplate ); + if ( !document().isNull() && !m_styleSheet.isNull() ) { + //document().removeStyleSheet(m_styleSheet); + //document().addStyleSheet(m_styleSheet); + } +} + +void SetupDisplay::setBackgroundColor() +{ + KreDisplayItem *item = *node_item_map->find( m_currNodeId ); + if ( KColorDialog::getColor( item->backgroundColor, view() ) == TQDialog::Accepted ) { + m_currentItem = item; + loadBackgroundColor(m_currNodeId,item->backgroundColor); + m_currentItem = 0; + + applyStylesheet(); + has_changes = true; + } +} + +void SetupDisplay::setBorder() +{ + KreDisplayItem *item = *node_item_map->find( m_currNodeId ); + BorderDialog borderDialog( item->border, view() ); + if ( borderDialog.exec() == TQDialog::Accepted ) { + m_currentItem = item; + loadBorder( m_currNodeId, borderDialog.border() ); + m_currentItem = 0; + + applyStylesheet(); + has_changes = true; + } +} + +void SetupDisplay::setColumns() +{ + KreDisplayItem *item = *node_item_map->find( m_currNodeId ); + int cols = TQInputDialog::getInteger( TQString::null, i18n("Select the number of columns to use:"), item->columns, 1, 100, 1, 0, view() ); + if ( cols > 0 ) { + m_currentItem = item; + loadColumns( m_currNodeId, cols ); + m_currentItem = 0; + + loadTemplate( m_activeTemplate ); + has_changes = true; + } +} + +void SetupDisplay::setTextColor() +{ + KreDisplayItem *item = *node_item_map->find( m_currNodeId ); + if ( KColorDialog::getColor( item->textColor, view() ) == TQDialog::Accepted ) { + m_currentItem = item; + loadTextColor(m_currNodeId,item->textColor); + m_currentItem = 0; + + applyStylesheet(); + has_changes = true; + } +} + +void SetupDisplay::setShown( int id ) +{ + KreDisplayItem *item = *node_item_map->find( m_currNodeId ); + emit itemVisibilityChanged( item, !popup->isItemChecked( id ) ); + + m_currentItem = item; + loadVisibility(m_currNodeId,!popup->isItemChecked( id )); + m_currentItem = 0; + + applyStylesheet(); + has_changes = true; +} + +void SetupDisplay::setFont() +{ + KreDisplayItem *item = *node_item_map->find( m_currNodeId ); + if ( TDEFontDialog::getFont( item->font, false, view() ) == TQDialog::Accepted ) { + m_currentItem = item; + loadFont(m_currNodeId,item->font); + m_currentItem = 0; + + applyStylesheet(); + has_changes = true; + } +} + +void SetupDisplay::setAlignment( TQAction *action ) +{ + KreDisplayItem *item = *node_item_map->find( m_currNodeId ); + + //TODO: isn't there a simpler way to do this... + //preserve non-horizontal alignment flags + if ( item->alignment & TQt::AlignRight ) + item->alignment ^= TQt::AlignRight; + if ( item->alignment & TQt::AlignHCenter ) + item->alignment ^= TQt::AlignHCenter; + if ( item->alignment & TQt::AlignLeft ) + item->alignment ^= TQt::AlignLeft; + + if ( action->text() == i18n( "Center" ) ) + item->alignment |= TQt::AlignHCenter; + else if ( action->text() == i18n( "Left" ) ) + item->alignment |= TQt::AlignLeft; + else if ( action->text() == i18n( "Right" ) ) + item->alignment |= TQt::AlignRight; + + m_currentItem = item; + loadAlignment(m_currNodeId,item->alignment); + m_currentItem = 0; + + applyStylesheet(); + has_changes = true; +} + +void SetupDisplay::setItemShown( KreDisplayItem *item, bool visible ) +{ + item->show = visible; + + m_styleSheet.insertRule("."+item->nodeId+" { visibility:"+(item->show?"visible":"hidden")+" }",m_styleSheet.cssRules().length()); + applyStylesheet(); + + has_changes = true; +} + +void SetupDisplay::changeMade( void ) +{ + has_changes = true; +} + +#include "setupdisplay.moc" diff --git a/src/dialogs/setupdisplay.h b/src/dialogs/setupdisplay.h new file mode 100644 index 0000000..3edd092 --- /dev/null +++ b/src/dialogs/setupdisplay.h @@ -0,0 +1,148 @@ +/*************************************************************************** +* Copyright (C) 2003 by Jason Kivlighn * +* ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef SETUPDISPLAY_H +#define SETUPDISPLAY_H + +#include <tdehtml_part.h> +#include <dom/html_element.h> +#include <dom/css_stylesheet.h> + +#include <tqdom.h> +#include <tqwidget.h> +#include <tqmap.h> + +#include "datablocks/recipe.h" +#include "datablocks/kreborder.h" +#include "klomanager.h" + +#include <math.h> + +class TDEPopupMenu; + +class TQAction; +class TQLabel; +class TQWidget; + +class StyleSheet; + +class KreDisplayItem +{ +public: + KreDisplayItem( const TQString &id, const TQString &name ); + + void clear(); + + TQString nodeId; + TQString name; + KreBorder border; + int alignment; + int columns; + bool show; + TQColor backgroundColor; + TQColor textColor; + TQFont font; +}; + +typedef TQMap< KreDisplayItem*, unsigned int > PropertiesMap; + +/** @brief A TDEHTMLPart for editing specific CSS properties + * + * Set up the items of a recipe for display. + * + * @author Jason Kivlighn + */ +class SetupDisplay : public TDEHTMLPart, protected KLOManager +{ + TQ_OBJECT + +public: + SetupDisplay( const Recipe &, TQWidget *parent ); + ~SetupDisplay(); + + enum Properties { None = 0, BackgroundColor = 1, TextColor = 2, Font = 4, Visibility = 8, Alignment = 32, Columns = 64, Border = 128 }; + + void saveLayout( const TQString & ); + void loadLayout( const TQString & ); + void loadTemplate( const TQString &filename ); + + bool hasChanges() const + { + return has_changes; + } + + void setItemShown( KreDisplayItem *item, bool visible ); + + const PropertiesMap properties() const + { + return * box_properties; + } + + void reload(); + +signals: + void itemVisibilityChanged( KreDisplayItem *, bool ); + +protected: + virtual void begin (const KURL &url=KURL(), int xOffset=0, int yOffset=0); + + virtual void loadBackgroundColor( const TQString &obj, const TQColor& ); + virtual void loadFont( const TQString &obj, const TQFont& ); + virtual void loadTextColor( const TQString &obj, const TQColor& ); + virtual void loadVisibility( const TQString &obj, bool ); + virtual void loadAlignment( const TQString &obj, int ); + virtual void loadBorder( const TQString &obj, const KreBorder& ); + virtual void loadColumns( const TQString &obj, int ); + + virtual void beginObject( const TQString &obj ); + virtual void endObject(); + +protected slots: + void nodeClicked(const TQString &url,const TQPoint &point); + void changeMade(); + + //slots to set properties of item boxes + void setBackgroundColor(); + void setBorder(); + void setColumns(); + void setTextColor(); + void setFont(); + void setShown( int id ); + void setAlignment( TQAction * ); + +private: + PropertiesMap *box_properties; + TQMap<TQString, KreDisplayItem*> *node_item_map; + + bool has_changes; + + // Methods + void applyStylesheet(); + void loadPageLayout( const TQDomElement &tag ); + void loadHTMLView( const TQString &templateFile = TQString::null, const TQString &styleFile = TQString::null ); + + void createItem( const TQString &id, const TQString &name, unsigned int properties ); + + //the name of the element under the mouse on a right-click + TQString m_currNodeId; + + //the item corresponding to the current point of processing the KLO + KreDisplayItem *m_currentItem; + + TDEPopupMenu *popup; + DOM::CSSStyleSheet m_styleSheet; + Recipe m_sample; + + TQString m_activeTemplate; + TQString m_activeStyle; +}; + +#endif //SETUPDISPLAY_H + diff --git a/src/dialogs/shoppinglistdialog.cpp b/src/dialogs/shoppinglistdialog.cpp new file mode 100644 index 0000000..a0ac00f --- /dev/null +++ b/src/dialogs/shoppinglistdialog.cpp @@ -0,0 +1,267 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "shoppinglistdialog.h" + +#include <tdelocale.h> +#include <tdeconfig.h> +#include <kcursor.h> +#include <kdialog.h> +#include <tdeglobal.h> +#include <tdeapplication.h> + +#include "backends/recipedb.h" +#include "refineshoppinglistdialog.h" +#include "datablocks/recipelist.h" +#include "widgets/recipelistview.h" +#include "recipefilter.h" +#include "recipeactionshandler.h" + +/** A simple listview to accept dropping a RecipeItemDrag */ +class ShoppingListView : public TDEListView +{ +public: + ShoppingListView( TQWidget *parent ) : TDEListView( parent ) + {} + +protected: + bool acceptDrag( TQDropEvent *event ) const + { + return RecipeItemDrag::canDecode( event ); + } + + TQDragObject *dragObject() + { + RecipeListItem * item = dynamic_cast<RecipeListItem*>( selectedItem() ); + if ( item != 0 ) { + + RecipeItemDrag * obj = new RecipeItemDrag( item, this, "Recipe drag item" ); + /*const TQPixmap *pm = item->pixmap(0); + if( pm ) + obj->setPixmap( *pm );*/ + return obj; + } + return 0; + } +}; + + +ShoppingListDialog::ShoppingListDialog( TQWidget *parent, RecipeDB *db ) : TQWidget( parent ) +{ + // Store pointer to database + database = db; + + // Design dialog + layout = new TQGridLayout( this, 2, 2, KDialog::marginHint(), KDialog::spacingHint() ); + + recipeListView = new KreListView ( this, i18n( "Full recipe list" ), true, 1 ); + layout->addWidget( recipeListView, 0, 0 ); + listview = new RecipeListView( recipeListView, database ); + listview->setSizePolicy( TQSizePolicy::Ignored, TQSizePolicy::MinimumExpanding ); + listview->setDragEnabled( true ); + listview->setAcceptDrops( true ); + listview->setDropVisualizer( false ); + connect( recipeListView, TQ_SIGNAL( textChanged(const TQString&) ), TQ_SLOT( ensurePopulated() ) ); + connect( listview, TQ_SIGNAL( dropped( TDEListView*, TQDropEvent*, TQListViewItem* ) ), + this, TQ_SLOT( slotDropped( TDEListView*, TQDropEvent*, TQListViewItem* ) ) ); + recipeListView->setListView( listview ); + recipeListView->setCustomFilter( new RecipeFilter( recipeListView->listView() ), TQ_SLOT( filter( const TQString & ) ) ); + recipeListView->setSizePolicy( TQSizePolicy::Minimum, TQSizePolicy::MinimumExpanding ); + + TQBoxLayout* vboxl = new TQVBoxLayout( KDialog::spacingHint() ); + TDEIconLoader il; + addRecipeButton = new TQPushButton( this ); + addRecipeButton->setIconSet( il.loadIconSet( "forward", TDEIcon::Small ) ); + addRecipeButton->setFixedSize( TQSize( 32, 32 ) ); + addRecipeButton->setFlat( true ); + vboxl->addWidget( addRecipeButton ); + + removeRecipeButton = new TQPushButton( this ); + removeRecipeButton->setIconSet( il.loadIconSet( "back", TDEIcon::Small ) ); + removeRecipeButton->setFixedSize( TQSize( 32, 32 ) ); + removeRecipeButton->setFlat( true ); + vboxl->addWidget( removeRecipeButton ); + vboxl->addStretch(); + + layout->addItem( vboxl, 0, 1 ); + + shopRecipeListView = new KreListView ( this, i18n("Shopping List") ); + ShoppingListView *slistview = new ShoppingListView( shopRecipeListView ); + slistview->setSizePolicy( TQSizePolicy::Ignored, TQSizePolicy::MinimumExpanding ); + slistview->setDragEnabled( true ); + slistview->setAcceptDrops( true ); + slistview->setDropVisualizer( false ); + connect( slistview, TQ_SIGNAL( dropped( TDEListView*, TQDropEvent*, TQListViewItem* ) ), + this, TQ_SLOT( slotDropped( TDEListView*, TQDropEvent*, TQListViewItem* ) ) ); + shopRecipeListView->setListView( slistview ); + layout->addWidget( shopRecipeListView, 0, 2 ); + + shopRecipeListView->listView() ->addColumn( i18n( "Recipe Title" ) ); + + TDEConfig *config = TDEGlobal::config(); + config->setGroup( "Advanced" ); + bool show_id = config->readBoolEntry( "ShowID", false ); + shopRecipeListView->listView() ->addColumn( i18n( "Id" ), show_id ? -1 : 0 ); + + shopRecipeListView->listView() ->setSorting( -1 ); + shopRecipeListView->setSizePolicy( TQSizePolicy::Minimum, TQSizePolicy::MinimumExpanding ); + shopRecipeListView->listView() ->setAllColumnsShowFocus( true ); + + buttonBar = new TQHBox( this, "buttonBar" ); + layout->addMultiCellWidget( buttonBar, 1, 1, 0, 2 ); + + layout->setColStretch( 0, 1 ); + layout->setColStretch( 1, 0 ); + layout->setColStretch( 2, 1 ); + + okButton = new TQPushButton( buttonBar, "okButton" ); + okButton->setText( i18n( "&OK" ) ); + TQPixmap pm = il.loadIcon( "ok", TDEIcon::NoGroup, 16 ); + okButton->setIconSet( pm ); + + //buttonBar->layout()->addItem( new TQSpacerItem( 10,10, TQSizePolicy::MinimumExpanding, TQSizePolicy::Fixed ) ); + + clearButton = new TQPushButton( buttonBar, "clearButton" ); + clearButton->setText( i18n( "Clear" ) ); + pm = il.loadIcon( "edit-clear", TDEIcon::NoGroup, 16 ); + clearButton->setIconSet( pm ); + + //Takes care of all recipe actions and provides a popup menu to 'recipeListView' + actionHandler = new RecipeActionsHandler( recipeListView->listView(), database, RecipeActionsHandler::ExpandAll | RecipeActionsHandler::CollapseAll ); + + // Connect signals & slots + connect( addRecipeButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( addRecipe() ) ); + connect( removeRecipeButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( removeRecipe() ) ); + connect( okButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( showShoppingList() ) ); + connect( clearButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( clear() ) ); +} + +ShoppingListDialog::~ShoppingListDialog() +{} + +void ShoppingListDialog::ensurePopulated() +{ + listview->populateAll(); +} + +void ShoppingListDialog::createShopping( const RecipeList &rlist ) +{ + clear(); + RecipeList::const_iterator it; + for ( it = rlist.begin(); it != rlist.end(); ++it ) { + new RecipeListItem( shopRecipeListView->listView(), shopRecipeListView->listView() ->lastItem(), *it ); + } +} + +void ShoppingListDialog::reloadRecipeList( ReloadFlags flag ) +{ + ( ( RecipeListView* ) recipeListView->listView() ) ->reload( flag ); +} + +void ShoppingListDialog::reload( ReloadFlags flag ) +{ + reloadRecipeList ( flag ); // Missing: check if there's non-existing recipes in the list now, and if so, delete. +} + +void ShoppingListDialog::addRecipe( void ) +{ + TQPtrList<TQListViewItem> items = recipeListView->listView()->selectedItems(); + + TQPtrListIterator<TQListViewItem> it(items); + TQListViewItem *item; + while ( (item = it.current()) != 0 ) { + addRecipe( item ); + ++it; + } +} + +void ShoppingListDialog::addRecipe( TQListViewItem *item ) +{ + if ( item ) { + if ( item->rtti() == 1000 ) { + RecipeListItem * recipe_it = ( RecipeListItem* ) item; + + Recipe r; + r.title = recipe_it->title(); + r.recipeID = recipe_it->recipeID(); + ( void ) new RecipeListItem( shopRecipeListView->listView(), r ); + } + } +} + +void ShoppingListDialog::removeRecipe( void ) +{ + TQListViewItem * it; + it = shopRecipeListView->listView() ->selectedItem(); + if ( it ) + delete it; +} + +void ShoppingListDialog::showShoppingList( void ) +{ + // Store the recipe list in ElementList object first + ElementList recipeList; + RecipeListItem *it; + for ( it = ( RecipeListItem* ) shopRecipeListView->listView() ->firstChild();it;it = ( RecipeListItem* ) it->nextSibling() ) { + Element newEl; + newEl.id = it->recipeID(); + newEl.name = it->title(); // Storing the title is not necessary, but do it just in case it's used later on + recipeList.append( newEl ); + } + + RefineShoppingListDialog refineDialog( this, database, recipeList ); + refineDialog.exec(); +} + +void ShoppingListDialog::addRecipeToShoppingList( int recipeID ) +{ + Recipe r; + r.title = database->recipeTitle( recipeID ); + r.recipeID = recipeID; + + new RecipeListItem( shopRecipeListView->listView(), r ); +} + +void ShoppingListDialog::clear() +{ + shopRecipeListView->listView() ->clear(); +} + +void ShoppingListDialog::slotDropped( TDEListView *list, TQDropEvent *e, TQListViewItem * /*after*/ ) +{ + Recipe r; + RecipeListItem *item = new RecipeListItem( recipeListView->listView(), r ); // needs parent, use this temporarily + if ( !RecipeItemDrag::decode( e, *item ) ) { + delete item; + return ; + } + + if ( list == shopRecipeListView->listView() ) { + addRecipe( item ); + } + //find and delete the item if we just dropped onto the recipe list from the shopping list + else if ( list == recipeListView->listView() && e->source() == shopRecipeListView->listView() ) { + TQListViewItemIterator list_it = TQListViewItemIterator( shopRecipeListView->listView() ); + while ( list_it.current() ) { + if ( ( ( RecipeListItem* ) list_it.current() ) ->recipeID() == item->recipeID() ) { + delete list_it.current(); + break; + } + list_it++; + } + } + + delete item; + item = 0; // not needed anymore +} + +#include "shoppinglistdialog.moc" diff --git a/src/dialogs/shoppinglistdialog.h b/src/dialogs/shoppinglistdialog.h new file mode 100644 index 0000000..cc8dc43 --- /dev/null +++ b/src/dialogs/shoppinglistdialog.h @@ -0,0 +1,76 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef SHOPPINGLISTDIALOG_H +#define SHOPPINGLISTDIALOG_H + +#include <tqhbox.h> +#include <tqlayout.h> +#include <kpushbutton.h> +#include <kiconloader.h> + +#include "widgets/krelistview.h" +#include "widgets/dblistviewbase.h" + +class RecipeDB; +class RecipeList; +class ShoppingListViewDialog; +class RecipeActionsHandler; +class RecipeListView; + +/** +@author Unai Garro +*/ + +class ShoppingListDialog: public TQWidget +{ + TQ_OBJECT +public: + + ShoppingListDialog( TQWidget *parent, RecipeDB *db ); + ~ShoppingListDialog(); + void reload( ReloadFlags flag = Load ); + void createShopping( const RecipeList &rlist ); + +private: + + // Internal variables + RecipeDB *database; + // Internal Methods + void reloadRecipeList( ReloadFlags flag = Load ); + void addRecipe( TQListViewItem *item ); + // Widgets + TQGridLayout* layout; + TQPushButton* addRecipeButton; + TQPushButton* removeRecipeButton; + KreListView* recipeListView; + KreListView* shopRecipeListView; + TQHBox *buttonBar; + TQPushButton* okButton; + TQPushButton* clearButton; + ShoppingListViewDialog *shoppingListDisplay; + RecipeActionsHandler *actionHandler; + RecipeListView *listview; + +public slots: + void addRecipeToShoppingList( int recipeID ); // Called from inputdialog-> through krecipesview + +private slots: + void addRecipe( void ); + void removeRecipe( void ); + void showShoppingList( void ); + void clear( void ); + void slotDropped( TDEListView *list, TQDropEvent *e, TQListViewItem *after ); + void ensurePopulated(); +}; + +#endif diff --git a/src/dialogs/shoppinglistviewdialog.cpp b/src/dialogs/shoppinglistviewdialog.cpp new file mode 100644 index 0000000..9099aa1 --- /dev/null +++ b/src/dialogs/shoppinglistviewdialog.cpp @@ -0,0 +1,102 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "shoppinglistviewdialog.h" +#include "datablocks/ingredientlist.h" +#include "datablocks/mixednumber.h" + +#include <tqpushbutton.h> + +#include <tdelocale.h> +#include <tdeapplication.h> +#include <tdeconfig.h> +#include <kiconloader.h> + +ShoppingListViewDialog::ShoppingListViewDialog( TQWidget *parent, const IngredientList &ingredientList ) + : KDialogBase( parent, "shoppingviewdialog", true, TQString::null, + KDialogBase::Close | KDialogBase::User1, KDialogBase::Close, + false, KStdGuiItem::print() ) +{ + // Design dialog + TQVBox *page = makeVBoxMainWidget(); + + shoppingListView = new TDEHTMLPart( page ); + + setInitialSize( TQSize(350, 450) ); + + connect ( this, TQ_SIGNAL( user1Clicked() ), this, TQ_SLOT( print() ) ); + connect ( this, TQ_SIGNAL( closeClicked() ), this, TQ_SLOT( accept() ) ); + + //---------- Sort the list -------- + IngredientList list_copy = ingredientList; + qHeapSort( list_copy ); + + //---------- Load the list -------- + display( list_copy ); +} + + +ShoppingListViewDialog::~ShoppingListViewDialog() +{} + +void ShoppingListViewDialog::display( const IngredientList &ingredientList ) +{ + TQString recipeHTML; + + // Create HTML Code + + // Headers + recipeHTML = TQString( "<html><head><title>%1</title></head><body>" ).arg( i18n( "Shopping List" ) ); + recipeHTML += "<center><div STYLE=\"width: 95%\">"; + recipeHTML += TQString( "<center><h1>%1</h1></center>" ).arg( i18n( "Shopping List" ) ); + + // Ingredient List + + recipeHTML += "<div STYLE=\"border:medium solid blue; width:95%\"><table cellspacing=0px width=100%><tbody>"; + bool counter = true; + + TDEConfig *config = TDEGlobal::config(); + config->setGroup( "Formatting" ); + + bool useAbbreviations = config->readBoolEntry("AbbreviateUnits"); + bool useFraction = config->readBoolEntry( "Fraction" ); + + for ( IngredientList::const_iterator ing_it = ingredientList.begin(); ing_it != ingredientList.end(); ++ing_it ) { + TQString color = ( counter ) ? "#CBCEFF" : "#BFC2F0"; + counter = !counter; + + MixedNumber::Format number_format = ( useFraction ) ? MixedNumber::MixedNumberFormat : MixedNumber::DecimalFormat; + TQString amount_str = MixedNumber( ( *ing_it ).amount ).toString( number_format ); + + TQString unit = ( *ing_it ).units.determineName( ( *ing_it ).amount + ( *ing_it ).amount_offset, useAbbreviations ); + + recipeHTML += TQString( "<tr bgcolor=\"%1\"><td>- %2:</td><td>%3 %4</td></tr>" ).arg( color ).arg( ( *ing_it ).name ).arg( amount_str ).arg( unit ); + } + recipeHTML += "</tbody></table></div>"; + // Close + recipeHTML += "</div></center></body></html>"; + + + // Display + shoppingListView->begin( KURL( "file:/tmp/" ) ); // Initialize to /tmp, where photos and logos are stored + shoppingListView->write( recipeHTML ); + shoppingListView->end(); + + +} + +void ShoppingListViewDialog::print() +{ + shoppingListView->view() ->print(); +} + +#include "shoppinglistviewdialog.moc" diff --git a/src/dialogs/shoppinglistviewdialog.h b/src/dialogs/shoppinglistviewdialog.h new file mode 100644 index 0000000..e080808 --- /dev/null +++ b/src/dialogs/shoppinglistviewdialog.h @@ -0,0 +1,46 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef SHOPPINGLISTVIEWDIALOG_H +#define SHOPPINGLISTVIEWDIALOG_H + +#include <tqlayout.h> +#include <tqvbox.h> +#include <tqwidget.h> + +#include <tdehtml_part.h> +#include <tdehtmlview.h> +#include <kdialogbase.h> + +class IngredientList; + +class ShoppingListViewDialog: public KDialogBase +{ + TQ_OBJECT + +public: + ShoppingListViewDialog( TQWidget *parent, const IngredientList &ingredientList ); + ~ShoppingListViewDialog(); + +public slots: + void print(); + +private: + + // Widgets + TDEHTMLPart *shoppingListView; + + // Internal Methods + void display( const IngredientList &ingredientList ); +}; + +#endif diff --git a/src/dialogs/similarcategoriesdialog.cpp b/src/dialogs/similarcategoriesdialog.cpp new file mode 100644 index 0000000..75d9721 --- /dev/null +++ b/src/dialogs/similarcategoriesdialog.cpp @@ -0,0 +1,383 @@ +/*************************************************************************** +* Copyright (C) 2003-2006 Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "similarcategoriesdialog.h" + +#include <tqvariant.h> +#include <tqpushbutton.h> +#include <tqcombobox.h> +#include <tqslider.h> +#include <tqlabel.h> +#include <tqheader.h> +#include <tqlayout.h> +#include <tqtooltip.h> +#include <tqwhatsthis.h> + +#include <tdelistview.h> +#include <tdelocale.h> +#include <tdemessagebox.h> +#include <klineedit.h> + +#include "widgets/categorycombobox.h" +#include "backends/recipedb.h" + +SimilarCategoriesDialog::SimilarCategoriesDialog( ElementList &list, TQWidget* parent ) + : TQDialog( parent, "SimilarCategoriesDialog", true ), + m_elementList(list) +{ + SimilarCategoriesDialogLayout = new TQVBoxLayout( this, 11, 6, "SimilarCategoriesDialogLayout"); + + layout6 = new TQHBoxLayout( 0, 0, 6, "layout6"); + + layout4 = new TQGridLayout( 0, 1, 1, 0, 6, "layout4"); + + categoriesBox = new KLineEdit( this ); + + layout4->addWidget( categoriesBox, 0, 1 ); + + thresholdSlider = new TQSlider( this, "thresholdSlider" ); + thresholdSlider->setValue( 40 ); + thresholdSlider->setOrientation( TQSlider::Horizontal ); + + layout4->addWidget( thresholdSlider, 1, 1 ); + + thresholdLabel = new TQLabel( this, "thresholdLabel" ); + + layout4->addWidget( thresholdLabel, 1, 0 ); + + categoryLabel = new TQLabel( this, "categoryLabel" ); + + layout4->addWidget( categoryLabel, 0, 0 ); + layout6->addLayout( layout4 ); + + layout5 = new TQVBoxLayout( 0, 0, 6, "layout5"); + + searchButton = new TQPushButton( this, "searchButton" ); + layout5->addWidget( searchButton ); + spacer4 = new TQSpacerItem( 20, 51, TQSizePolicy::Minimum, TQSizePolicy::Expanding ); + layout5->addItem( spacer4 ); + layout6->addLayout( layout5 ); + SimilarCategoriesDialogLayout->addLayout( layout6 ); + + layout9 = new TQHBoxLayout( 0, 0, 6, "layout9"); + + layout8 = new TQVBoxLayout( 0, 0, 6, "layout8"); + + allLabel = new TQLabel( this, "allLabel" ); + layout8->addWidget( allLabel ); + + allListView = new TDEListView( this, "allListView" ); + allListView->setSizePolicy( TQSizePolicy( (TQSizePolicy::SizeType)7, (TQSizePolicy::SizeType)7, 0, 1, allListView->sizePolicy().hasHeightForWidth() ) ); + layout8->addWidget( allListView ); + layout9->addLayout( layout8 ); + + layout1 = new TQVBoxLayout( 0, 0, 6, "layout1"); + + removeButton = new TQPushButton( this, "removeButton" ); + layout1->addWidget( removeButton ); + + addButton = new TQPushButton( this, "addButton" ); + layout1->addWidget( addButton ); + spacer1 = new TQSpacerItem( 20, 61, TQSizePolicy::Minimum, TQSizePolicy::Expanding ); + layout1->addItem( spacer1 ); + layout9->addLayout( layout1 ); + + layout7 = new TQVBoxLayout( 0, 0, 6, "layout7"); + + toMergeLabel = new TQLabel( this, "toMergeLabel" ); + layout7->addWidget( toMergeLabel ); + + toMergeListView = new TDEListView( this, "toMergeListView" ); + toMergeListView->setSizePolicy( TQSizePolicy( (TQSizePolicy::SizeType)7, (TQSizePolicy::SizeType)7, 0, 1, toMergeListView->sizePolicy().hasHeightForWidth() ) ); + layout7->addWidget( toMergeListView ); + layout9->addLayout( layout7 ); + SimilarCategoriesDialogLayout->addLayout( layout9 ); + + layout10 = new TQHBoxLayout( 0, 0, 6, "layout10"); + spacer2 = new TQSpacerItem( 310, 20, TQSizePolicy::Expanding, TQSizePolicy::Minimum ); + layout10->addItem( spacer2 ); + + mergeButton = new TQPushButton( this, "mergeButton" ); + layout10->addWidget( mergeButton ); + + cancelButton = new TQPushButton( this, "cancelButton" ); + layout10->addWidget( cancelButton ); + SimilarCategoriesDialogLayout->addLayout( layout10 ); + languageChange(); + resize( TQSize(573, 429).expandedTo(minimumSizeHint()) ); + clearWState( WState_Polished ); + + connect( searchButton, TQ_SIGNAL(clicked()), this, TQ_SLOT(findMatches()) ); + connect( mergeButton, TQ_SIGNAL(clicked()), this, TQ_SLOT(mergeMatches()) ); + connect( cancelButton, TQ_SIGNAL(clicked()), this, TQ_SLOT(reject()) ); + connect( addButton, TQ_SIGNAL(clicked()), this, TQ_SLOT(addCategory()) ); + connect( removeButton, TQ_SIGNAL(clicked()), this, TQ_SLOT(removeCategory()) ); +} + +/* + * Destroys the object and frees any allocated resources + */ +SimilarCategoriesDialog::~SimilarCategoriesDialog() +{ + // no need to delete child widgets, TQt does it all for us +} + +/* + * Sets the strings of the subwidgets using the current + * language. + */ +void SimilarCategoriesDialog::languageChange() +{ + setCaption( i18n( "Similar Categories" ) ); + thresholdLabel->setText( i18n( "Threshold:" ) ); + categoryLabel->setText( i18n( "Category:" ) ); + searchButton->setText( i18n( "Search" ) ); + allLabel->setText( i18n( "Similar Categories:" ) ); + removeButton->setText( i18n( "<<" ) ); + addButton->setText( i18n( ">>" ) ); + toMergeLabel->setText( i18n( "Categories to Merge:" ) ); + mergeButton->setText( i18n( "Merge" ) ); + cancelButton->setText( i18n( "Cancel" ) ); + + allListView->addColumn( i18n( "Category" ) ); + //allListView->addColumn( i18n( "Id" ) ); + toMergeListView->addColumn( i18n( "Category" ) ); + //toMergeListView->addColumn( i18n( "Id" ) ); +} + +/*****************************************************/ +/*Function prototypes and libraries needed to compile*/ +/*****************************************************/ + +#include <stdlib.h> +#include <malloc.h> +#include <string.h> +int levenshtein_distance(const char *s,const char*t); +int minimum(int a,int b,int c); + +/****************************************/ +/*Implementation of Levenshtein distance*/ +/****************************************/ + +int levenshtein_distance(const char *s,const char*t) +/*Compute levenshtein distance between s and t*/ +{ + //Step 1 + int k,i,j,n,m,cost,*d,distance; + n=strlen(s); + m=strlen(t); + if(n!=0&&m!=0) + { + d=(int*)malloc((sizeof(int))*(m+1)*(n+1)); + m++; + n++; + //Step 2 + for(k=0;k<n;k++) + d[k]=k; + for(k=0;k<m;k++) + d[k*n]=k; + //Step 3 and 4 + for(i=1;i<n;i++) + for(j=1;j<m;j++) + { + //Step 5 + if(s[i-1]==t[j-1]) + cost=0; + else + cost=1; + //Step 6 + d[j*n+i]=minimum(d[(j-1)*n+i]+1,d[j*n+i-1]+1,d[(j-1)*n+i-1]+cost); + } + distance=d[n*m-1]; + free(d); + return distance; + } + else + return -1; //a negative return value means that one or both strings are empty. +} + +int minimum(int a,int b,int c) +/*Gets the minimum of three values*/ +{ + int min=a; + if(b<min) + min=b; + if(c<min) + min=c; + return min; +} + +/** @return an array of adjacent letter pairs contained in the input string */ +TQStringList letterPairs(const TQString& str) { + int numPairs = str.length()-1; + TQStringList pairs; + for (int i=0; i<numPairs; i++) { + pairs << str.mid(i,2); + } + return pairs; +} + +/** @return an ArrayList of 2-character Strings. */ +TQValueList<TQStringList> wordLetterPairs(const TQString &str) { + TQValueList<TQStringList> allPairs; + // Tokenize the string and put the tokens/words into an array + TQStringList words = TQStringList::split("\\s",str); + // For each word + for (uint w=0; w < words.count(); w++) { + // Find the pairs of characters + TQStringList pairsInWord = letterPairs(words[w]); + for (uint p=0; p < pairsInWord.count(); p++) { + allPairs.append(pairsInWord[p]); + } + } + return allPairs; +} + +/** @return lexical similarity value in the range [0,1] */ +double compareStrings(const TQString &str1, const TQString &str2) { + TQValueList<TQStringList> pairs1 = wordLetterPairs(str1.upper()); + TQValueList<TQStringList> pairs2 = wordLetterPairs(str2.upper()); + int intersection = 0; + int size_union = pairs1.count() + pairs2.count(); + for (uint i=0; i<pairs1.count(); i++) { + TQStringList pair1=pairs1[i]; + for(uint j=0; j<pairs2.count(); j++) { + TQStringList pair2=pairs2[j]; + if (pair1 == pair2) { + intersection++; + pairs2.remove( pairs2.at(j) ); + break; + } + } + } + return (2.0*intersection)/size_union; +} + + + +#include <kdebug.h> + +#if 0 +void RecipeActionsHandler::mergeSimilar() +{ + TQPtrList<TQListViewItem> items = parentListView->selectedItems(); + if ( items.count() > 1 ) + KMessageBox::sorry( kapp->mainWidget(), i18n("Please select only one category."), TQString::null ); + else if ( items.count() == 1 && items.at(0)->rtti() == 1001 ) { + CategoryListItem * cat_it = ( CategoryListItem* ) items.at(0); + TQString name = cat_it->categoryName(); + const double max_allowed_distance = 0.60; + const int length = name.length(); + ElementList categories; + database->loadCategories( &categories ); + + ElementList matches; + for ( ElementList::const_iterator it = categories.begin(); it != categories.end(); ++it ) { + #if 0 + if ( levenshtein_distance(name.latin1(),(*it).name.latin1())/double(TQMAX(length,(*it).name.length())) >= max_allowed_distance ) { + #else + if ( compareStrings(name,(*it).name) >= max_allowed_distance ) { + #endif + kdDebug()<<(*it).name<<" matches"<<endl; + if ( cat_it->categoryId() != (*it).id ) + matches.append(*it); + } + } + + + for ( ElementList::const_iterator it = categories.begin(); it != categories.end(); ++it ) { + database->mergeCategories(cat_it->categoryId(),(*it).id); + } + + } + else //either nothing was selected or a recipe was selected + KMessageBox::sorry( kapp->mainWidget(), i18n("No recipes selected."), i18n("Edit Recipe") ); +} +#endif + +void SimilarCategoriesDialog::findMatches() +{ + allListView->clear(); + toMergeListView->clear(); + + const double threshold = (100 - thresholdSlider->value())/100.0; + const TQString name = categoriesBox->text(); + + for ( ElementList::const_iterator it = m_elementList.begin(); it != m_elementList.end(); ++it ) { + //kdDebug()<<(*it).name<<" (result/threshold): "<<compareStrings(name,(*it).name)<<"/"<<threshold<<endl; + #if 0 + if ( levenshtein_distance(name.latin1(),(*it).name.latin1())/double(TQMAX(length,(*it).name.length())) >= max_allowed_distance ) { + #else + if ( compareStrings(name,(*it).name) >= threshold ) { + #endif + kdDebug()<<(*it).name<<" matches"<<endl; + //if ( id != (*it).id ) { + (void) new TQListViewItem(allListView,(*it).name,TQString::number((*it).id)); + (void) new TQListViewItem(toMergeListView,(*it).name,TQString::number((*it).id)); + //} + } + } +} + +void SimilarCategoriesDialog::mergeMatches() +{ + if ( !toMergeListView->firstChild() ) { + KMessageBox::sorry( this, i18n("No categories selected to merge."), TQString::null ); + return; + } + + //const int id = categoriesBox->id(categoriesBox->currentItem()); + //for ( TQListViewItem *item = toMergeListView->firstChild(); item; item = item->nextSibling() ) { + // m_database->mergeCategories(id,item->text(1).toInt()); + //} + + allListView->clear(); + //toMergeListView->clear(); + + TQDialog::accept(); +} + +TQValueList<int> SimilarCategoriesDialog::matches() const +{ + TQValueList<int> ids; + for ( TQListViewItem *item = toMergeListView->firstChild(); item; item = item->nextSibling() ) { + ids << item->text(1).toInt(); + } + + return ids; +} + +TQString SimilarCategoriesDialog::element() const +{ + return categoriesBox->text(); +} + +void SimilarCategoriesDialog::addCategory() +{ + TQListViewItem *item = allListView->selectedItem(); + if ( item ) + { + //make sure it isn't already in the list + for ( TQListViewItem *search_it = toMergeListView->firstChild(); search_it; search_it = search_it->nextSibling() ) { + if ( search_it->text(0) == item->text(0) ) + return; + } + + (void) new TQListViewItem(toMergeListView,item->text(0),item->text(1)); + } +} + +void SimilarCategoriesDialog::removeCategory() +{ + TQListViewItem *item = toMergeListView->selectedItem(); + if ( item ) + delete item; +} + +#include "similarcategoriesdialog.moc" diff --git a/src/dialogs/similarcategoriesdialog.h b/src/dialogs/similarcategoriesdialog.h new file mode 100644 index 0000000..67d6d03 --- /dev/null +++ b/src/dialogs/similarcategoriesdialog.h @@ -0,0 +1,84 @@ +/*************************************************************************** +* Copyright (C) 2003-2006 Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef SIMILARCATEGORIESDIALOG_H +#define SIMILARCATEGORIESDIALOG_H + +#include <tqvariant.h> +#include <tqdialog.h> +#include <tqvaluevector.h> + +#include "datablocks/elementlist.h" + +class TQVBoxLayout; +class TQHBoxLayout; +class TQGridLayout; +class TQSpacerItem; +class TQSlider; +class TQLabel; +class TQPushButton; +class TQListViewItem; + +class TDEListView; +class KLineEdit; + +class RecipeDB; + +class SimilarCategoriesDialog : public TQDialog +{ +TQ_OBJECT + +public: + SimilarCategoriesDialog( ElementList &, TQWidget* parent = 0 ); + ~SimilarCategoriesDialog(); + + TQValueList<int> matches() const; + TQString element() const; + + KLineEdit* categoriesBox; + TQSlider* thresholdSlider; + TQLabel* thresholdLabel; + TQLabel* categoryLabel; + TQPushButton* searchButton; + TQLabel* allLabel; + TDEListView* allListView; + TQPushButton* removeButton; + TQPushButton* addButton; + TQLabel* toMergeLabel; + TDEListView* toMergeListView; + TQPushButton* mergeButton; + TQPushButton* cancelButton; + +protected: + TQVBoxLayout* SimilarCategoriesDialogLayout; + TQHBoxLayout* layout6; + TQGridLayout* layout4; + TQVBoxLayout* layout5; + TQSpacerItem* spacer4; + TQHBoxLayout* layout9; + TQVBoxLayout* layout8; + TQVBoxLayout* layout1; + TQSpacerItem* spacer1; + TQVBoxLayout* layout7; + TQHBoxLayout* layout10; + TQSpacerItem* spacer2; + +protected slots: + virtual void languageChange(); + void findMatches(); + void mergeMatches(); + void addCategory(); + void removeCategory(); + +private: + ElementList m_elementList; + +}; + +#endif // SIMILARCATEGORIESDIALOG_H diff --git a/src/dialogs/unitsdialog.cpp b/src/dialogs/unitsdialog.cpp new file mode 100644 index 0000000..d52328a --- /dev/null +++ b/src/dialogs/unitsdialog.cpp @@ -0,0 +1,211 @@ +/*************************************************************************** +* Copyright (C) 2003-2004 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include <tqlayout.h> +#include <tqtabwidget.h> + +#include "unitsdialog.h" +#include "createelementdialog.h" +#include "dependanciesdialog.h" +#include "backends/recipedb.h" +#include "widgets/conversiontable.h" +#include "widgets/unitlistview.h" + +#include <tdeapplication.h> +#include <kdebug.h> +#include <kdialog.h> +#include <tdelocale.h> +#include <tdemessagebox.h> +#include <kprogress.h> + +UnitsDialog::UnitsDialog( TQWidget *parent, RecipeDB *db ) : TQWidget( parent ) +{ + + // Store pointer to database + database = db; + + // Design dialog + TQHBoxLayout* page_layout = new TQHBoxLayout( this, KDialog::marginHint(), KDialog::spacingHint() ); + + TQTabWidget *tabWidget = new TQTabWidget( this ); + + TQWidget *unitTab = new TQWidget( tabWidget ); + TQHBoxLayout* layout = new TQHBoxLayout( unitTab, KDialog::marginHint(), KDialog::spacingHint() ); + + unitListView = new StdUnitListView( unitTab, database, true ); + layout->addWidget( unitListView ); + + TQVBoxLayout* vboxl = new TQVBoxLayout( KDialog::spacingHint() ); + newUnitButton = new TQPushButton( unitTab ); + newUnitButton->setText( i18n( "Create ..." ) ); + newUnitButton->setFlat( true ); + vboxl->addWidget( newUnitButton ); + + removeUnitButton = new TQPushButton( unitTab ); + removeUnitButton->setText( i18n( "Delete" ) ); + removeUnitButton->setFlat( true ); + vboxl->addWidget( removeUnitButton ); + vboxl->addStretch(); + layout->addLayout( vboxl ); + + tabWidget->insertTab( unitTab, i18n( "Units" ) ); + + massConversionTable = new ConversionTable( tabWidget, 1, 1 ); + tabWidget->insertTab( massConversionTable, i18n( "Mass Conversions" ) ); + + volumeConversionTable = new ConversionTable( tabWidget, 1, 1 ); + tabWidget->insertTab( volumeConversionTable, i18n( "Volume Conversions" ) ); + + page_layout->addWidget( tabWidget ); + + // Connect signals & slots + connect( newUnitButton, TQ_SIGNAL( clicked() ), unitListView, TQ_SLOT( createNew() ) ); + connect( removeUnitButton, TQ_SIGNAL( clicked() ), unitListView, TQ_SLOT( remove() ) ); + connect( massConversionTable, TQ_SIGNAL( ratioChanged( int, int, double ) ), this, TQ_SLOT( saveRatio( int, int, double ) ) ); + connect( massConversionTable, TQ_SIGNAL( ratioRemoved( int, int ) ), this, TQ_SLOT( removeRatio( int, int ) ) ); + connect( volumeConversionTable, TQ_SIGNAL( ratioChanged( int, int, double ) ), this, TQ_SLOT( saveRatio( int, int, double ) ) ); + connect( volumeConversionTable, TQ_SIGNAL( ratioRemoved( int, int ) ), this, TQ_SLOT( removeRatio( int, int ) ) ); + + //TODO: I'm too lazy right now, so do a complete reload to keep in sync with db + connect( database, TQ_SIGNAL( unitCreated( const Unit& ) ), this, TQ_SLOT( loadConversionTables() ) ); + connect( database, TQ_SIGNAL( unitRemoved( int ) ), this, TQ_SLOT( loadConversionTables() ) ); + + //this is for the above TODO, but it still has some bugs to be worked out + //connect(database,TQ_SIGNAL(unitCreated(const Element&)),conversionTable,TQ_SLOT(unitCreated(const Element&))); + //connect(database,TQ_SIGNAL(unitRemoved(int)),conversionTable,TQ_SLOT(unitRemoved(int))); + + //Populate data into the table + loadConversionTables(); + + //FIXME: We've got some sort of build issue... we get undefined references to CreateElementDialog without this dummy code here + CreateElementDialog d( this, "" ); +} + +UnitsDialog::~UnitsDialog() +{} + +void UnitsDialog::reload( ReloadFlags flag ) +{ + unitListView->reload( flag ); + loadConversionTables(); +} + +void UnitsDialog::loadConversionTables( void ) +{ + loadConversionTable( massConversionTable, Unit::Mass ); + loadConversionTable( volumeConversionTable, Unit::Volume ); +} + +void UnitsDialog::loadConversionTable( ConversionTable *table, Unit::Type type ) +{ + UnitList unitList; + database->loadUnits( &unitList, type ); + + TQStringList unitNames; + IDList unitIDs; // We need to store these in the table, so rows and cols are identified by unitID, not name. + table->clear(); + for ( UnitList::const_iterator unit_it = unitList.begin(); unit_it != unitList.end(); ++unit_it ) { + unitNames.append( ( *unit_it ).name ); + unitIDs.append( ( *unit_it ).id ); // append the element + } + + // Resize the table + table->resize( unitNames.count(), unitNames.count() ); + + // Set the table labels, and id's + table->setRowLabels( unitNames ); + table->setColumnLabels( unitNames ); + table->setUnitIDs( unitIDs ); + + + // Load and Populate the data into the table + UnitRatioList ratioList; + database->loadUnitRatios( &ratioList, type ); + for ( UnitRatioList::const_iterator ratio_it = ratioList.begin(); ratio_it != ratioList.end(); ++ratio_it ) { + table->setRatio( ( *ratio_it ).uID1, ( *ratio_it ).uID2, ( *ratio_it ).ratio ); + } +} + +void UnitsDialog::saveRatio( int r, int c, double value ) +{ + ConversionTable *conversionTable = (ConversionTable*)sender(); + UnitRatio ratio; + + ratio.uID1 = conversionTable->getUnitID( r ); + ratio.uID2 = conversionTable->getUnitID( c ); + ratio.ratio = value; + database->saveUnitRatio( &ratio ); + + UnitRatio reverse_ratio; + reverse_ratio.uID1 = ratio.uID2; + reverse_ratio.uID2 = ratio.uID1; + reverse_ratio.ratio = 1.0 / ratio.ratio; + database->saveUnitRatio( &reverse_ratio ); + conversionTable->setRatio( reverse_ratio ); + +#if 0 + UnitRatioList ratioList; + database->loadUnitRatios( &ratioList, Unit::Mass ); + + saveAllRatios( ratioList ); +#endif +} + +void UnitsDialog::removeRatio( int r, int c ) +{ + ConversionTable *conversionTable = (ConversionTable*)sender(); + database->removeUnitRatio( conversionTable->getUnitID( r ), conversionTable->getUnitID( c ) ); +} + +void UnitsDialog::saveAllRatios( UnitRatioList &ratioList ) +{ +#if 0 + ConversionTable *conversionTable = massConversionTable; + + KProgressDialog progress_dialog( this, "progress_dialog", i18n( "Finding Unit Ratios" ), TQString::null, true ); + progress_dialog.progressBar() ->setTotalSteps( ratioList.count() * ratioList.count() ); + + for ( UnitRatioList::const_iterator current_it = ratioList.begin(); current_it != ratioList.end(); ++current_it ) { + UnitRatio current_ratio = *current_it; + for ( UnitRatioList::const_iterator ratio_it = ratioList.begin(); ratio_it != ratioList.end(); ++ratio_it ) { + if ( progress_dialog.wasCancelled() ) + return ; + + progress_dialog.progressBar() ->advance( 1 ); + kapp->processEvents(); + + UnitRatio new_ratio; + new_ratio.uID1 = current_ratio.uID1; + new_ratio.uID2 = ( *ratio_it ).uID2; + new_ratio.ratio = ( *ratio_it ).ratio * current_ratio.ratio; + + if ( ratioList.contains( new_ratio ) ) + continue; + + if ( ( ( *ratio_it ).uID1 == current_ratio.uID2 ) && ( ( *ratio_it ).uID2 != current_ratio.uID1 ) ) { + UnitRatio reverse_ratio; + reverse_ratio.uID1 = new_ratio.uID2; + reverse_ratio.uID2 = new_ratio.uID1; + reverse_ratio.ratio = 1.0 / new_ratio.ratio; + + database->saveUnitRatio( &new_ratio ); + database->saveUnitRatio( &reverse_ratio ); + conversionTable->setRatio( new_ratio ); + conversionTable->setRatio( reverse_ratio ); + //ratioList.append(new_ratio); ratioList.append(reverse_ratio); + } + } + } +#endif +} + +#include "unitsdialog.moc" diff --git a/src/dialogs/unitsdialog.h b/src/dialogs/unitsdialog.h new file mode 100644 index 0000000..0810656 --- /dev/null +++ b/src/dialogs/unitsdialog.h @@ -0,0 +1,65 @@ +/*************************************************************************** +* Copyright (C) 2003-2004 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef UNITSDIALOG_H +#define UNITSDIALOG_H +#include <tqwidget.h> +#include <tdelistview.h> +#include <tqpushbutton.h> + +#include "datablocks/unitratiolist.h" +#include "datablocks/unit.h" +#include "widgets/dblistviewbase.h" + +class RecipeDB; +class ConversionTable; +class StdUnitListView; + +/** +@author Unai Garro +*/ +class UnitsDialog: public TQWidget +{ + TQ_OBJECT +public: + UnitsDialog( TQWidget *parent, RecipeDB *db ); + ~UnitsDialog(); + virtual TQSize sizeHint () const + { + return TQSize( 300, 200 ); + } + +public slots: + void reload( ReloadFlags flag = Load ); + +private: + // Widgets + StdUnitListView *unitListView; + ConversionTable *massConversionTable; + ConversionTable *volumeConversionTable; + TQPushButton *newUnitButton; + TQPushButton *removeUnitButton; + + // Internal methods + void saveAllRatios( UnitRatioList &ratioList ); + bool checkBounds( const TQString &name ); + + // Internal Variables + RecipeDB *database; +private slots: + void loadConversionTables(); + void loadConversionTable( ConversionTable*, Unit::Type ); + void saveRatio( int r, int c, double value ); + void removeRatio( int r, int c ); +}; + +#endif diff --git a/src/dialogs/usdadatadialog.cpp b/src/dialogs/usdadatadialog.cpp new file mode 100644 index 0000000..fd771c6 --- /dev/null +++ b/src/dialogs/usdadatadialog.cpp @@ -0,0 +1,200 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "usdadatadialog.h" + +#include <kdebug.h> +#include <kstandarddirs.h> +#include <klineedit.h> +#include <tdelistview.h> +#include <tdelocale.h> +#include <tdemessagebox.h> + +#include <tqfile.h> +#include <tqlabel.h> +#include <tqlayout.h> +#include <tqpushbutton.h> +#include <tqtextstream.h> +#include <tqvbox.h> + +#include "backends/recipedb.h" +#include "backends/usda_property_data.h" +#include "backends/usda_unit_data.h" +#include "widgets/krelistview.h" +#include "datablocks/weight.h" + +USDADataDialog::USDADataDialog( const Element &ing, RecipeDB *db, TQWidget *parent ) + : KDialogBase( parent, "usdaDataDialog", true, TQString::null, + KDialogBase::Ok | KDialogBase::Cancel, KDialogBase::Ok ), + ingredient( ing ), + database( db ) +{ + setCaption( TQString( i18n( "Load ingredient properties for: \"%1\"" ) ).arg( ingredient.name ) ); + + TQVBox *page = makeVBoxMainWidget(); + + setButtonText( KDialogBase::Ok, i18n( "&Load" ) ); + + KreListView *krelistview = new KreListView( page, TQString::null, true, 0 ); + + listView = krelistview->listView(); + listView->addColumn( i18n( "USDA Ingredient" ) ); + listView->addColumn( i18n( "Id" ) ); + listView->setAllColumnsShowFocus( true ); + + loadDataFromFile(); + + connect( listView, TQ_SIGNAL( doubleClicked( TQListViewItem*, const TQPoint &, int ) ), this, TQ_SLOT( slotOk() ) ); +} + +USDADataDialog::~USDADataDialog() +{} + +void USDADataDialog::loadDataFromFile() +{ + TQString abbrev_file = locate( "appdata", "data/abbrev.txt" ); + if ( abbrev_file.isEmpty() ) { + kdDebug() << "Unable to find abbrev.txt data file." << endl; + return ; + } + + TQFile file( abbrev_file ); + if ( !file.open( IO_ReadOnly ) ) { + kdDebug() << "Unable to open data file: " << abbrev_file << endl; + return ; + } + + int index = 0; + TQTextStream stream( &file ); + while ( !stream.atEnd() ) { + TQString line = stream.readLine(); + if ( line.isEmpty() ) { + continue; + } + + TQStringList fields = TQStringList::split( "^", line, true ); + loaded_data << fields; + + TQString ing_id = fields[ 0 ].mid( 1, fields[ 1 ].length() - 2 ); + TQString ing_name = fields[ 1 ].mid( 1, fields[ 1 ].length() - 2 ); + ( void ) new TQListViewItem( listView, ing_name, TQString::number( index ) ); //using an index instead of the actual id will help find the data later + + index++; + } +} + +void USDADataDialog::slotOk() +{ + TQListViewItem * item = listView->selectedItem(); + if ( item ) { + int index = item->text( 1 ).toInt(); + TQStringList data = loaded_data[ index ]; + + int grams_id = database->findExistingUnitByName( "g" ); //get this id because all data is given per gram + if ( grams_id == -1 ) { + //FIXME: take advantage of abbreviations + Unit unit("g","g"); + unit.type = Unit::Mass; + database->createNewUnit( unit ); + grams_id = database->lastInsertID(); + } + else { + Unit unit = database->unitName(grams_id); + if ( unit.type != Unit::Mass ) { + unit.type = Unit::Mass; + database->modUnit( unit ); + } + } + + IngredientPropertyList property_list; + database->loadProperties( &property_list ); + IngredientPropertyList existing_ing_props; + database->loadProperties( &existing_ing_props, ingredient.id ); + + int i = 0; + for ( TQStringList::const_iterator it = data.at( 2 ); !property_data_list[ i ].name.isEmpty(); ++it, ++i ) { + int property_id = property_list.findByName( property_data_list[ i ].name ); + if ( property_id == -1 ) { + database->addProperty( property_data_list[ i ].name, property_data_list[ i ].unit ); + property_id = database->lastInsertID(); + } + + double amount = ( *it ).toDouble() / 100.0; //data givin per 100g so divide by 100 to get the amount in 1 gram + + if ( existing_ing_props.find( property_id ) != existing_ing_props.end() ) //property already added to ingredient, so just update + database->changePropertyAmountToIngredient( ingredient.id, property_id, amount, grams_id ); + else + database->addPropertyToIngredient( ingredient.id, property_id, amount, grams_id ); + } + + i+=2; + + int i_initial = i; + WeightList weights = database->ingredientWeightUnits( ingredient.id ); + for ( ; i < i_initial+3; ++i ) { + Weight w; + w.weight = data[i].toDouble(); + + i++; + + TQString amountAndWeight = data[i].mid( 1, data[i].length() - 2 ); + if ( !amountAndWeight.isEmpty() ) { + int spaceIndex = amountAndWeight.find(" "); + w.perAmount = amountAndWeight.left(spaceIndex).toDouble(); + + TQString perAmountUnit = amountAndWeight.right(amountAndWeight.length()-spaceIndex-1); + if ( !parseUSDAUnitAndPrep( perAmountUnit, w.perAmountUnit, w.prepMethod ) ) + continue; + + int unitID = database->findExistingUnitByName( w.perAmountUnit ); + if ( unitID == -1 ) { + for ( int i = 0; unit_data_list[ i ].name; ++i ) { + if ( w.perAmountUnit == unit_data_list[ i ].name || w.perAmountUnit == unit_data_list[ i ].plural ) { + database->createNewUnit( Unit(unit_data_list[ i ].name,unit_data_list[ i ].plural) ); + } + } + + unitID = database->lastInsertID(); + } + w.perAmountUnitID = unitID; + + if ( !w.prepMethod.isEmpty() ) { + int prepID = database->findExistingPrepByName( w.prepMethod ); + if ( prepID == -1 ) { + database->createNewPrepMethod( w.prepMethod ); + prepID = database->lastInsertID(); + } + w.prepMethodID = prepID; + } + + bool exists = false; + for ( WeightList::const_iterator it = weights.begin(); it != weights.end(); ++it ) { + if ( (*it).perAmountUnitID == w.perAmountUnitID && (*it).prepMethodID == w.prepMethodID ) { + exists = true; + break; + } + } + if ( exists ) + continue; + + w.weightUnitID = grams_id; + w.ingredientID = ingredient.id; + database->addIngredientWeight( w ); + } + } + + accept(); + } + else + reject(); +} + + +#include "usdadatadialog.moc" diff --git a/src/dialogs/usdadatadialog.h b/src/dialogs/usdadatadialog.h new file mode 100644 index 0000000..b96d9de --- /dev/null +++ b/src/dialogs/usdadatadialog.h @@ -0,0 +1,47 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef USDADATADIALOG_H +#define USDADATADIALOG_H + +#include <kdialogbase.h> + +#include <tqstringlist.h> +#include <tqvaluelist.h> + +#include "datablocks/element.h" + +class RecipeDB; + +class TDEListView; + +class USDADataDialog : public KDialogBase +{ + TQ_OBJECT + +public: + USDADataDialog( const Element &, RecipeDB *database, TQWidget *parent = 0 ); + ~USDADataDialog(); + +private: + void loadDataFromFile(); + + TDEListView *listView; + + Element ingredient; + RecipeDB *database; + + TQValueList<TQStringList> loaded_data; + +private slots: + void slotOk(); +}; + +#endif //USDADATADIALOG_H diff --git a/src/exporters/Makefile.am b/src/exporters/Makefile.am new file mode 100644 index 0000000..fcedc46 --- /dev/null +++ b/src/exporters/Makefile.am @@ -0,0 +1,13 @@ +INCLUDES = -I$(srcdir) -I$(srcdir)/.. $(all_includes) + +METASOURCES = AUTO +noinst_LTLIBRARIES = libkrecipesexporters.la + +libkrecipesexporters_la_LDFLAGS = $(KDE_RPATH) $(all_libraries) + +libkrecipesexporters_la_SOURCES = kreexporter.cpp baseexporter.cpp cookmlexporter.cpp \ + recipemlexporter.cpp mmfexporter.cpp htmlexporter.cpp plaintextexporter.cpp \ + rezkonvexporter.cpp htmlbookexporter.cpp + +noinst_HEADERS = kreexporter.h baseexporter.h cookmlexporter.h \ + recipemlexporter.h mmfexporter.h htmlexporter.h plaintextexporter.h diff --git a/src/exporters/baseexporter.cpp b/src/exporters/baseexporter.cpp new file mode 100644 index 0000000..60dc5a1 --- /dev/null +++ b/src/exporters/baseexporter.cpp @@ -0,0 +1,172 @@ +/*************************************************************************** +* Copyright (C) 2003-2005 by * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "baseexporter.h" + +#include <tqfile.h> +#include <tqfileinfo.h> + +#include <tdeaboutdata.h> +#include <kdebug.h> +#include <tdelocale.h> +#include <tdeglobal.h> +#include <tdemessagebox.h> +#include <ktar.h> +#include <kstandarddirs.h> + +#include "backends/recipedb.h" + +BaseExporter::BaseExporter( const TQString& _filename, const TQString &format ) : + file( 0 ), + tar_file( 0 ), + filename( _filename ), + m_progress_dlg( 0 ), + compress(false) +{ + //automatically append extension + TQString extension = format.right( format.length()-2 ); + if ( filename.right( extension.length() ) != extension ) + filename += "." + extension; +} + +BaseExporter::~BaseExporter() +{ + delete file; + delete tar_file; +} + +int BaseExporter::headerFlags() const +{ + return RecipeDB::None; +} + +void BaseExporter::setCompressed( bool b ) +{ + compress = b; +} + +void BaseExporter::exporter( const TQValueList<int> &ids, RecipeDB *database, KProgressDialog *progress_dlg ) +{ + m_progress_dlg = progress_dlg; + + if ( createFile() ) + saveToFile( ids, database ); + else + kdDebug() << "no output file has been selected for export." << endl; +} + +void BaseExporter::exporter( int id, RecipeDB *database, KProgressDialog *progress_dlg ) +{ + TQValueList<int> single_recipe_list; + single_recipe_list << id ; + exporter( single_recipe_list, database, progress_dlg ); +} + +void BaseExporter::writeStream( TQTextStream &stream, const RecipeList &recipe_list ) +{ + stream << createHeader(recipe_list); + stream << createContent(recipe_list); + stream << createFooter(); +} + +bool BaseExporter::createFile() +{ + if ( file ) + return true; + + if ( !filename.isEmpty() ) { + if ( compress ) { + tar_file = new KTar( filename, "application/x-gzip" ); + TQFileInfo fi( filename ); + file = new TQFile( locateLocal( "tmp",fi.fileName()+"ml" ) ); + } + else + file = new TQFile(filename); + + return (file != 0); + } + else + return false; +} + +TQString BaseExporter::fileName() const +{ + return filename; +} + +void BaseExporter::saveToFile( const TQValueList<int> &ids, RecipeDB *database ) +{ + if ( file->open( IO_WriteOnly ) ) { + if ( m_progress_dlg ) + m_progress_dlg->progressBar()->setTotalSteps( ids.count()/progressInterval() ); + + TQValueList<int> ids_copy = ids; + TQTextStream stream( file ); + stream.setEncoding( TQTextStream::UnicodeUTF8 ); + + RecipeList recipe_list; + if ( headerFlags() != RecipeDB::None ) { + database->loadRecipes( &recipe_list, headerFlags(), ids ); + } + stream << createHeader( recipe_list ); + + recipe_list.clear(); + for ( uint i = 0; i < ids.count(); i += progressInterval() ) { + TQValueList<int> sub_list; + for ( int sub_i = 0; sub_i < progressInterval(); ++sub_i ) { + if ( ids_copy.count() == 0 ) break; + + sub_list << *ids_copy.begin(); + ids_copy.remove( ids_copy.begin() ); + } + + RecipeList recipe_list; + database->loadRecipes( &recipe_list, supportedItems(), sub_list ); + + TQString content = createContent( recipe_list ); + if ( !content.isEmpty() ) + stream << content; + + if ( m_progress_dlg && m_progress_dlg->wasCancelled() ) + break; + + if ( m_progress_dlg ) { + m_progress_dlg->progressBar()->advance( progressInterval() ); + kapp->processEvents(); + } + } + + stream << createFooter(); + + if ( tar_file && tar_file->open( IO_WriteOnly ) ) { + //close, which flushes the buffer, and then open for reading + file->close(); + file->open( IO_ReadOnly ); + + TQFileInfo fi( file->name() ); + TQByteArray data = file->readAll(); + tar_file->writeFile( fi.fileName(), fi.owner(), fi.group(), data.size(), data ); + tar_file->close(); + } + + file->close(); + } +} + +TQString BaseExporter::krecipes_version() const +{ + TDEInstance * this_instance = TDEGlobal::instance(); + if ( this_instance && this_instance->aboutData() ) + return this_instance->aboutData() ->version(); + + return TQString::null; //Oh, well. We couldn't get the version. +} + diff --git a/src/exporters/baseexporter.h b/src/exporters/baseexporter.h new file mode 100644 index 0000000..d814fb4 --- /dev/null +++ b/src/exporters/baseexporter.h @@ -0,0 +1,95 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef BASEEXPORTER_H +#define BASEEXPORTER_H + +#include <tqstringlist.h> + +#include <tdeapplication.h> +#include <kprogress.h> + +#include "datablocks/recipelist.h" + +class TQFile; + +class KTar; + +class RecipeDB; + +class BaseExporter +{ +public: + BaseExporter( const TQString &file, const TQString &ext ); + virtual ~BaseExporter(); + + /** Subclasses must report which items it is able to work with. + * These should be or'ed together items from RecipeDB::RecipeItems + */ + virtual int supportedItems() const = 0; + + /** Export the recipes with the given ids to the file specified in the constructor. + * Optionally, a progress dialog may be given to specify the progress made. + */ + void exporter( const TQValueList<int> &ids, RecipeDB *database, KProgressDialog * = 0 ); + + /** Convenience function for the above, which exports a single recipe. */ + void exporter( int id, RecipeDB *database, KProgressDialog * = 0 ); + + /** Returns the actual filename that will be written to during the export. + * Note that this can differ somewhat from the filename passed in the + * constructor. + */ + TQString fileName() const; + + /** Write the given recipe list to a text stream. + * This can be used to export recipes without use of the database. + */ + void writeStream( TQTextStream &, const RecipeList & ); + +protected: + virtual TQString createContent( const RecipeList & ) = 0; + virtual TQString createFooter(){ return TQString(); } + virtual TQString createHeader( const RecipeList & ){ return TQString(); } + + /** The number of recipes to load into memory at once. This many recipes will be + * loaded from the database, processed, and then another batch of this many will be + * processed until all recipes are exported. + */ + virtual int progressInterval() const { return 50; } + + /** Extra RecipeDB::RecipeItems that a subclass requires when creating a file's header. + * For example, the Krecipes file format requires writing the category hierarchy in the header, + * so it's exporter adds RecipeDB::Categories. + */ + virtual int headerFlags() const; + + /** Make generated file a gzipped tarball */ + void setCompressed( bool ); + + /** Attempt to return the version of the application via + * TDEGlobal::instance()->aboutData()->version() + * This can be used by exporters to put the version of the app exporting the file. + */ + TQString krecipes_version() const; + +private: + bool createFile(); + void saveToFile( const TQValueList<int> &ids, RecipeDB *database ); + + TQFile* file; + KTar *tar_file; + TQString filename; + KProgressDialog *m_progress_dlg; + bool compress; +}; + +#endif //BASEEXPORTER_H diff --git a/src/exporters/cookmlexporter.cpp b/src/exporters/cookmlexporter.cpp new file mode 100644 index 0000000..f8f9d40 --- /dev/null +++ b/src/exporters/cookmlexporter.cpp @@ -0,0 +1,121 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "cookmlexporter.h" + +#include <tqbuffer.h> +#include <tqdom.h> +#include <tqimage.h> +#include <tqpixmap.h> +#include <tqfile.h> + +#include <tdeconfig.h> +#include <kdebug.h> +#include <tdelocale.h> +#include <tdetempfile.h> +#include <kmdcodec.h> +#include <tdeglobal.h> +#include <kstandarddirs.h> + +#include "backends/recipedb.h" + +CookMLExporter::CookMLExporter( const TQString& filename, const TQString &format ) : + BaseExporter( filename, format ) +{} + + +CookMLExporter::~CookMLExporter() +{} + +int CookMLExporter::supportedItems() const +{ + return RecipeDB::All ^ RecipeDB::Ratings; +} + +TQString CookMLExporter::createHeader( const RecipeList& ) +{ + TQString xml = "<?xml version=\"1.0\" encoding=\"UTF-8\" ?>\n"; + xml += "<!DOCTYPE cookml PUBLIC \"-\" \"cookml.dtd\">"; + xml += "<cookml version=\"1.0.13\" prog=\"Krecipes\" progver=\""+krecipes_version()+"\">"; + return xml; +} + +TQString CookMLExporter::createFooter() +{ + return "</cookml>"; +} + +TQString CookMLExporter::createContent( const RecipeList& recipes ) +{ + TQString xml; + TQDomDocument doc; + + RecipeList::const_iterator recipe_it; + for ( recipe_it = recipes.begin(); recipe_it != recipes.end(); ++recipe_it ) { + TQDomElement recipe_tag = doc.createElement( "recipe" ); + recipe_tag.setAttribute( "lang", ( TDEGlobal::locale() ) ->language() ); + + //cookml_tag.appendChild( recipe_tag ); + + TQDomElement head_tag = doc.createElement( "head" ); + head_tag.setAttribute( "title", ( *recipe_it ).title ); + head_tag.setAttribute( "servingqty", ( *recipe_it ).yield.amount ); + head_tag.setAttribute( "servingtype", ( *recipe_it ).yield.type ); + head_tag.setAttribute( "rid", "" ); //FIXME:what's this...recipe ID?? + recipe_tag.appendChild( head_tag ); + + for ( ElementList::const_iterator cat_it = ( *recipe_it ).categoryList.begin(); cat_it != ( *recipe_it ).categoryList.end(); ++cat_it ) { + TQDomElement cat_tag = doc.createElement( "cat" ); + cat_tag.appendChild( doc.createTextNode( ( *cat_it ).name ) ); + head_tag.appendChild( cat_tag ); + } + + for ( ElementList::const_iterator author_it = ( *recipe_it ).authorList.begin(); author_it != ( *recipe_it ).authorList.end(); ++author_it ) { + TQDomElement sourceline_tag = doc.createElement( "sourceline" ); + sourceline_tag.appendChild( doc.createTextNode( ( *author_it ).name ) ); + head_tag.appendChild( sourceline_tag ); + } + + TQDomElement picbin_tag = doc.createElement( "picbin" ); + picbin_tag.setAttribute( "format", "JPG" ); + + TQByteArray data; + TQBuffer buffer( data ); + buffer.open( IO_WriteOnly ); + TQImageIO iio( &buffer, "JPEG" ); + iio.setImage( ( *recipe_it ).photo.convertToImage() ); + iio.write(); + + picbin_tag.appendChild( doc.createTextNode( KCodecs::base64Encode( data, true ) ) ); + head_tag.appendChild( picbin_tag ); + + TQDomElement part_tag = doc.createElement( "part" ); + for ( IngredientList::const_iterator ing_it = ( *recipe_it ).ingList.begin(); ing_it != ( *recipe_it ).ingList.end(); ++ing_it ) { + TQDomElement ingredient_tag = doc.createElement( "ingredient" ); + ingredient_tag.setAttribute( "qty", TQString::number( ( *ing_it ).amount ) ); + ingredient_tag.setAttribute( "unit", ( ( *ing_it ).amount > 1 ) ? ( *ing_it ).units.plural : ( *ing_it ).units.name ); + ingredient_tag.setAttribute( "item", ( *ing_it ).name ); + ingredient_tag.setAttribute( "preparation", ( *ing_it ).prepMethodList.join(",") ); + part_tag.appendChild( ingredient_tag ); + } + recipe_tag.appendChild( part_tag ); + + TQDomElement preparation_tag = doc.createElement( "preparation" ); + recipe_tag.appendChild( preparation_tag ); + + TQDomElement text_tag = doc.createElement( "text" ); + preparation_tag.appendChild( text_tag ); + text_tag.appendChild( doc.createTextNode( ( *recipe_it ).instructions ) ); + + xml += recipe_tag.text(); + } + + return xml; +} diff --git a/src/exporters/cookmlexporter.h b/src/exporters/cookmlexporter.h new file mode 100644 index 0000000..7e8d99a --- /dev/null +++ b/src/exporters/cookmlexporter.h @@ -0,0 +1,38 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef COOKMLEXPORTER_H +#define COOKMLEXPORTER_H + +#include "baseexporter.h" + +/** + * Export class for the Meal-Master file format + * @author Jason Kivlighn + * + * Note: This format does not handle all the properties of recipes. + * Data lost in export to this format include: + * ---none?--- + */ +class CookMLExporter : public BaseExporter +{ +public: + CookMLExporter( const TQString&, const TQString& ); + virtual ~CookMLExporter(); + + virtual int supportedItems() const; + +protected: + virtual TQString createContent( const RecipeList& ); + virtual TQString createHeader( const RecipeList& ); + virtual TQString createFooter(); +}; + +#endif //COOKMLEXPORTER_H diff --git a/src/exporters/htmlbookexporter.cpp b/src/exporters/htmlbookexporter.cpp new file mode 100644 index 0000000..93440a5 --- /dev/null +++ b/src/exporters/htmlbookexporter.cpp @@ -0,0 +1,160 @@ +/*************************************************************************** +* Copyright (C) 2006 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "htmlbookexporter.h" + +#include <tqfile.h> +#include <tqstylesheet.h> + +#include <kdebug.h> + +#include "backends/recipedb.h" +#include "datablocks/categorytree.h" + +HTMLBookExporter::HTMLBookExporter( CategoryTree *categories, const TQString& basedir, const TQString &format ) : + HTMLExporter( basedir+"/index", format ), m_categories(categories), m_basedir(basedir) +{ +} + +HTMLBookExporter::~HTMLBookExporter() +{ +} + +int HTMLBookExporter::headerFlags() const +{ + return RecipeDB::Categories | RecipeDB::Title; +} + +TQString HTMLBookExporter::createContent( const RecipeList& recipes ) +{ + RecipeList::const_iterator recipe_it; + for ( recipe_it = recipes.begin(); recipe_it != recipes.end(); ++recipe_it ) { + for ( ElementList::const_iterator cat_it = ( *recipe_it ).categoryList.begin(); cat_it != ( *recipe_it ).categoryList.end(); ++cat_it ) { + TQMap<TQString,TQTextStream*>::iterator stream_it = fileMap.find( (*cat_it).name ); + (**stream_it) << "<br /><br />"; + (**stream_it) << TQString("<a name=\""+(*recipe_it).title+"\" />"); + (**stream_it) << HTMLExporter::createContent(*recipe_it); + (**stream_it) << TQString("[ <a href=\"#top\">Top</a> ]"); + (**stream_it) << TQString("[ <a href=\"index.html\">Back</a> ]"); + (**stream_it) << "<br /><br />"; + } + } + + return TQString::null; +} + +TQString HTMLBookExporter::createHeader( const RecipeList &list ) +{ + TQString output = HTMLExporter::createHeader(list); + + TQString catLinks; + TQTextStream catLinksStream(&catLinks,IO_WriteOnly); + createCategoryStructure(catLinksStream,list); + + return output+"<h1>Krecipes Recipes</h1><div>"+catLinks+"</li></ul></div>"; +} + +TQString HTMLBookExporter::createFooter() +{ + TQMap<TQString,TQTextStream*>::const_iterator it; + for ( it = fileMap.begin(); it != fileMap.end(); ++it ) { + (**it) << HTMLExporter::createFooter(); + + (*it)->device()->close(); + + //does it matter the order of deletion here? + TQIODevice *file = (*it)->device(); + delete *it; + delete file; + } + + TQString output = HTMLExporter::createFooter(); + return output; +} + +void HTMLBookExporter::createCategoryStructure( TQTextStream &xml, const RecipeList &recipes ) +{ + TQValueList<int> categoriesUsed; + for ( RecipeList::const_iterator recipe_it = recipes.begin(); recipe_it != recipes.end(); ++recipe_it ) { + for ( ElementList::const_iterator cat_it = ( *recipe_it ).categoryList.begin(); cat_it != ( *recipe_it ).categoryList.end(); ++cat_it ) { + TQMap<TQString,TQTextStream*>::iterator stream_it = fileMap.find( (*cat_it).name ); + if ( categoriesUsed.find( ( *cat_it ).id ) == categoriesUsed.end() ) { + categoriesUsed << ( *cat_it ).id; + + TQString catPageName = m_basedir+"/"+escape((*cat_it).name)+".html"; + TQFile *catPage = new TQFile( catPageName ); + catPage->open( IO_WriteOnly ); + TQTextStream *stream = new TQTextStream( catPage ); + stream->setEncoding( TQTextStream::UnicodeUTF8 ); + (*stream) << HTMLExporter::createHeader(recipes); + (*stream) << TQString("<a name=\"top\" />"); + (*stream) << "<h1>"<<(*cat_it).name<<"</h1>"; + + stream_it = fileMap.insert((*cat_it).name,stream); + } + (**stream_it) << TQString("[ <a href=\"#" + (*recipe_it).title + "\">" + (*recipe_it).title + "</a> ]"); + } + } + + if ( !categoriesUsed.empty() ) { + //only keep the relevant category structure + removeIfUnused( categoriesUsed, m_categories ); + + xml << "<ul>\n"; + writeCategoryStructure( xml, m_categories ); + xml << "</ul>\n"; + } +} + +bool HTMLBookExporter::removeIfUnused( const TQValueList<int> &cat_ids, CategoryTree *parent, bool parent_should_show ) +{ + for ( CategoryTree * it = parent->firstChild(); it; it = it->nextSibling() ) { + if ( cat_ids.find( it->category.id ) != cat_ids.end() ) { + parent_should_show = true; + removeIfUnused( cat_ids, it, true ); //still recurse, but doesn't affect 'parent' + } + else { + bool result = removeIfUnused( cat_ids, it ); + if ( parent_should_show == false ) + parent_should_show = result; + } + } + + if ( !parent_should_show && parent->category.id != -1 ) { + //FIXME: CategoryTree is broken when deleting items + //delete parent; + + parent->category.id = -2; //temporary workaround + } + + return parent_should_show; +} + +void HTMLBookExporter::writeCategoryStructure( TQTextStream &xml, const CategoryTree *categoryTree ) +{ + if ( categoryTree->category.id != -2 ) { + if ( categoryTree->category.id != -1 ) { + TQString catPageName = TQStyleSheet::escape(categoryTree->category.name)+".html"; + + xml << "\t<li>\n\t\t<a href=\""+catPageName+"\">"+TQStyleSheet::escape( categoryTree->category.name ).replace("\"",""") + "</a>\n"; + } + + for ( CategoryTree * child_it = categoryTree->firstChild(); child_it; child_it = child_it->nextSibling() ) { + if ( categoryTree->parent() != 0 ) + xml << "<ul>\n"; + writeCategoryStructure( xml, child_it ); + if ( categoryTree->parent() != 0 ) + xml << "</ul>\n"; + } + + if ( categoryTree->category.id != -1 ) + xml << "\t</li>\n"; + } +} diff --git a/src/exporters/htmlbookexporter.h b/src/exporters/htmlbookexporter.h new file mode 100644 index 0000000..d8b5884 --- /dev/null +++ b/src/exporters/htmlbookexporter.h @@ -0,0 +1,52 @@ +/*************************************************************************** +* Copyright (C) 2006 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef HTMLBOOKEXPORTER_H +#define HTMLBOOKEXPORTER_H + +#include <tqmap.h> +#include <tqvaluelist.h> + +#include "baseexporter.h" +#include "htmlexporter.h" + +class RecipeDB; +class CategoryTree; + +/** + * Exports a given recipe list as HTML + * @author Jason Kivlighn + */ +class HTMLBookExporter : public HTMLExporter +{ +public: + HTMLBookExporter( CategoryTree *categories, const TQString&, const TQString& ); + virtual ~HTMLBookExporter(); + +protected: + virtual TQString createContent( const RecipeList & ); + virtual TQString createHeader( const RecipeList & ); + virtual TQString createFooter(); + + virtual int headerFlags() const; + +private: + void createCategoryStructure( TQTextStream &xml, const RecipeList &recipes ); + bool removeIfUnused( const TQValueList<int> &cat_ids, CategoryTree *parent, bool parent_should_show = false ); + void writeCategoryStructure( TQTextStream &xml, const CategoryTree *categoryTree ); + + TQMap<TQString,TQTextStream*> fileMap; + + RecipeDB *database; + CategoryTree *m_categories; + TQString m_basedir; +}; + +#endif //HTMLBOOKEXPORTER_H diff --git a/src/exporters/htmlexporter.cpp b/src/exporters/htmlexporter.cpp new file mode 100644 index 0000000..b4c44f4 --- /dev/null +++ b/src/exporters/htmlexporter.cpp @@ -0,0 +1,570 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Jason Kivlighn ([email protected]) * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* * +* Copyright (C) 2006 Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "htmlexporter.h" + +#include <tqptrdict.h> +#include <tqimage.h> +#include <tqfileinfo.h> +#include <tqdir.h> +#include <tqstylesheet.h> //for TQStyleSheet::escape() to escape for HTML +#include <dom/dom_element.h> +#include <tqpainter.h> +#include <tqfileinfo.h> + +#include <tdeconfig.h> +#include <kdebug.h> +#include <tdelocale.h> +#include <tdeglobal.h> +#include <tdehtml_part.h> +#include <tdehtmlview.h> +#include <kprogress.h> +#include <kstandarddirs.h> +#include <kurl.h> +#include <kiconloader.h> + +#include "datablocks/mixednumber.h" +#include "backends/recipedb.h" +#include "dialogs/setupdisplay.h" +#include "image.h" +#include "krepagelayout.h" + +#include <cmath> //for ceil() + +HTMLExporter::HTMLExporter( const TQString& filename, const TQString &format ) : + BaseExporter( filename, format ) +{ + TDEConfig *config = TDEGlobal::config(); + config->setGroup( "Page Setup" ); + + //let's do everything we can to be sure at least some layout is loaded + TQString template_filename = config->readEntry( "Template", locate( "appdata", "layouts/Default.template" ) ); + if ( template_filename.isEmpty() || !TQFile::exists( template_filename ) ) + template_filename = locate( "appdata", "layouts/Default.template" ); + kdDebug() << "Using template file: " << template_filename << endl; + + setTemplate( template_filename ); + + //let's do everything we can to be sure at least some layout is loaded + m_layoutFilename = config->readEntry( "Layout", locate( "appdata", "layouts/Default.klo" ) ); + if ( m_layoutFilename.isEmpty() || !TQFile::exists( m_layoutFilename ) ) + m_layoutFilename = locate( "appdata", "layouts/Default.klo" ); + kdDebug() << "Using layout file: " << m_layoutFilename << endl; +} + +HTMLExporter::~HTMLExporter() +{ +} + +void HTMLExporter::setTemplate( const TQString &filename ) +{ + TQFile templateFile( filename ); + if ( templateFile.open( IO_ReadOnly ) ) { + m_templateFilename = filename; + m_templateContent = TQString( templateFile.readAll() ); + } + else + kdDebug()<<"couldn't find/open template file"<<endl; +} + +void HTMLExporter::setStyle( const TQString &filename ) +{ + m_layoutFilename = filename; +} + +int HTMLExporter::supportedItems() const +{ + int items = RecipeDB::All ^ RecipeDB::Properties; + + TQMap<TQString,bool>::const_iterator it = m_visibilityMap.find("properties"); + if ( it == m_visibilityMap.end() || it.data() == true ) + items |= RecipeDB::Properties; + + return RecipeDB::All; +} + +TQString HTMLExporter::createContent( const Recipe& recipe ) +{ + TQString templateCopy = m_templateContent; + + storePhoto( recipe ); + + populateTemplate( recipe, templateCopy ); + return templateCopy; +} + +TQString HTMLExporter::createContent( const RecipeList& recipes ) +{ + TQString fileContent; + + RecipeList::const_iterator recipe_it; + for ( recipe_it = recipes.begin(); recipe_it != recipes.end(); ++recipe_it ) { + fileContent += createContent(*recipe_it); + } + + return fileContent; +} + +TQString HTMLExporter::createHeader( const RecipeList & ) +{ + m_visibilityMap.clear(); + m_columnsMap.clear(); + + TDEConfig *config = TDEGlobal::config(); + config->setGroup( "Page Setup" ); + + m_error = false; + + if ( m_templateContent.isEmpty() ) { + TQString errorStr = i18n("<html><body>\n" + "<p><b>Error: </b>Unable to find a layout file, which is" + " needed to view the recipe.</p>" + "<p>Krecipes was probably not properly installed.</p>" + "</body></html>"); + m_error = true; + return errorStr; + } + + TQFile layoutFile( m_layoutFilename ); + TQString error; int line; int column; + TQDomDocument doc; + if ( !doc.setContent( &layoutFile, &error, &line, &column ) ) { + kdDebug()<<"Unable to load style information. Will create HTML without it..."<<endl; + } + else + processDocument(doc); + + //put all the recipe photos into this directory + TQDir dir; + TQFileInfo fi(fileName()); + dir.mkdir( fi.dirPath(true) + "/" + fi.baseName() + "_photos" ); + + RecipeList::const_iterator recipe_it; + + TDELocale*loc = TDEGlobal::locale(); + TQString encoding = loc->encoding(); + + TQString output = "<html>"; + output += "<head>"; + output += "<meta name=\"lang\" content=\"" + loc->language() + "\">\n"; + output += "<meta http-equiv=\"Content-Type\" content=\"text/html; charset=utf-8\" />\n"; + output += TQString( "<title>%1</title>" ).arg( i18n( "Krecipes Recipes" ) ); + + output += "<style type=\"text/css\">\n"; + + TQString cssContent; + TQFileInfo info(m_templateFilename); + TQFile cssFile(info.dirPath(true) + "/" + info.baseName() + ".css"); + kdDebug()<<info.dirPath(true) + "/" + info.baseName() + ".css"<<endl; + if ( cssFile.open( IO_ReadOnly ) ) { + cssContent = TQString( cssFile.readAll() ); + } + output += cssContent; + + output += m_cachedCSS; + m_cachedCSS = TQString::null; + output += "</style>"; + output += "</head>"; + output += "<body class=\"background\">"; + + return output; +} + +void HTMLExporter::beginObject( const TQString &object ) +{ + m_cachedCSS += "."+object+" { \n"; +} + +void HTMLExporter::endObject() +{ + m_cachedCSS += " } \n"; +} + +void HTMLExporter::loadBackgroundColor( const TQString &/*object*/, const TQColor& color ) +{ + m_cachedCSS += bgColorAsCSS(color); +} + +void HTMLExporter::loadFont( const TQString &/*object*/, const TQFont& font ) +{ + m_cachedCSS += fontAsCSS(font); +} + +void HTMLExporter::loadTextColor( const TQString &/*object*/, const TQColor& color ) +{ + m_cachedCSS += textColorAsCSS(color); +} + +void HTMLExporter::loadVisibility( const TQString &object, bool visible ) +{ + m_cachedCSS += visibilityAsCSS(visible); + m_visibilityMap.insert(object,visible); +} + +void HTMLExporter::loadAlignment( const TQString &/*object*/, int alignment ) +{ + m_cachedCSS += alignmentAsCSS(alignment); +} + +void HTMLExporter::loadBorder( const TQString &/*object*/, const KreBorder& border ) +{ + m_cachedCSS += borderAsCSS(border); +} + +void HTMLExporter::loadColumns( const TQString & object, int cols ) +{ + m_columnsMap.insert(object,cols); +kdDebug()<<object<<" has "<<cols<<" columns"<<endl; +} + +TQString HTMLExporter::createFooter() +{ + if ( m_error ) + return TQString::null; + + return "</body></html>"; +} + +void HTMLExporter::storePhoto( const Recipe &recipe ) +{ + TQImage image; + TQString photo_name; + if ( recipe.photo.isNull() ) { + image = TQImage( defaultPhoto ); + photo_name = "default_photo"; + } + else { + image = recipe.photo.convertToImage(); + photo_name = TQString::number(recipe.recipeID); + } + + TQPixmap pm = image;//image.smoothScale( phwidth, 0, TQImage::ScaleMax ); + + TQFileInfo fi(fileName()); + TQString photo_path = fi.dirPath(true) + "/" + fi.baseName() + "_photos/" + photo_name + ".png"; + if ( !TQFile::exists( photo_path ) ) { + pm.save( photo_path, "PNG" ); + } +} + +TQString HTMLExporter::HTMLIfVisible( const TQString &name, const TQString &html ) +{ + TQMap<TQString,bool>::const_iterator it = m_visibilityMap.find(name); + if ( it == m_visibilityMap.end() || it.data() == true ) + return html; + else + return TQString::null; +} + +void HTMLExporter::populateTemplate( const Recipe &recipe, TQString &content ) +{ + TDEConfig * config = TDEGlobal::config(); + + //=======================TITLE======================// + content = content.replace("**TITLE**",HTMLIfVisible("title",recipe.title)); + + //=======================INSTRUCTIONS======================// + TQString instr_html = TQStyleSheet::escape( recipe.instructions ); + instr_html.replace( "\n", "<br />" ); + content = content.replace( "**INSTRUCTIONS**", HTMLIfVisible("instructions",instr_html) ); + + //=======================SERVINGS======================// + TQString yield_html = TQString( "<b>%1: </b>%2" ).arg( i18n( "Yield" ) ).arg( recipe.yield.toString() ); + content = content.replace( "**YIELD**", HTMLIfVisible("yield",yield_html) ); + + //=======================PREP TIME======================// + TQString preptime_html; + if ( !recipe.prepTime.isNull() && recipe.prepTime.isValid() ) + preptime_html = TQString( "<b>%1: </b>%2" ).arg( i18n( "Preparation Time" ) ).arg( recipe.prepTime.toString( "h:mm" ) ); + content = content.replace( "**PREP_TIME**", HTMLIfVisible("prep_time",preptime_html) ); + + //========================PHOTO========================// + TQString photo_name; + if ( recipe.photo.isNull() ) + photo_name = "default_photo"; + else + photo_name = TQString::number(recipe.recipeID); + + TQFileInfo fi(fileName()); + TQString image_url = fi.baseName() + "_photos/" + escape( photo_name ) + ".png"; + image_url = KURL::encode_string( image_url ); + content = content.replace( "**PHOTO**", HTMLIfVisible("photo",image_url) ); + + //=======================AUTHORS======================// + TQString authors_html; + + int counter = 0; + for ( ElementList::const_iterator author_it = recipe.authorList.begin(); author_it != recipe.authorList.end(); ++author_it ) { + if ( counter ) + authors_html += ", "; + authors_html += TQStyleSheet::escape( ( *author_it ).name ); + counter++; + } + if ( !authors_html.isEmpty() ) + authors_html.prepend( TQString( "<b>%1: </b>" ).arg( i18n( "Authors" ) ) ); + content = content.replace( "**AUTHORS**", HTMLIfVisible("authors",authors_html) ); + + //=======================CATEGORIES======================// + TQString categories_html; + + counter = 0; + for ( ElementList::const_iterator cat_it = recipe.categoryList.begin(); cat_it != recipe.categoryList.end(); ++cat_it ) { + if ( counter ) + categories_html += ", "; + categories_html += TQStyleSheet::escape( ( *cat_it ).name ); + counter++; + } + if ( !categories_html.isEmpty() ) + categories_html.prepend( TQString( "<b>%1: </b>" ).arg( i18n( "Categories" ) ) ); + + content = content.replace( "**CATEGORIES**", HTMLIfVisible("categories",categories_html) ); + + //=======================INGREDIENTS======================// + TQString ingredients_html; + config->setGroup( "Formatting" ); + + bool useAbbreviations = config->readBoolEntry("AbbreviateUnits"); + + MixedNumber::Format number_format = ( config->readBoolEntry( "Fraction" ) ) ? MixedNumber::MixedNumberFormat : MixedNumber::DecimalFormat; + + TQString ingredient_format = config->readEntry( "Ingredient", "%n%p: %a %u" ); + + TQMap<TQString,int>::const_iterator cols_it = m_columnsMap.find("ingredients"); + int cols = 1; + if ( cols_it != m_columnsMap.end() ) + cols = cols_it.data(); + int per_col = recipe.ingList.count() / cols; + if ( recipe.ingList.count() % cols != 0 ) //round up if division is not exact + per_col++; + + int count = 0; + IngredientList list_copy = recipe.ingList; //simple workaround until I fix iterating over the list dealing with groups + for ( IngredientList group_list = list_copy.firstGroup(); group_list.count() != 0; group_list = list_copy.nextGroup() ) { + TQString group = group_list[ 0 ].group; //just use the first's name... they're all the same + + bool loneHeader = false; + if ( count != 0 && count % per_col == 0 ) { + loneHeader = true; + if ( !group.isEmpty() ) + ingredients_html += "</ul>"; + ingredients_html.append("</ul></td><td valign=\"top\"><ul>"); + if ( !group.isEmpty() ) + ingredients_html += "<li style=\"page-break-after: avoid\">" + group + ":</li><ul>"; + } + else { + if ( !group.isEmpty() ) + ingredients_html += "<li style=\"page-break-after: avoid\">" + group + ":</li><ul>"; + } + + for ( IngredientList::const_iterator ing_it = group_list.begin(); ing_it != group_list.end(); ++ing_it, ++count ) { + if ( count != 0 && count % per_col == 0 && !loneHeader ) { + if ( !group.isEmpty() ) + ingredients_html += "</ul>"; + ingredients_html.append("</ul></td><td valign=\"top\"><ul>"); + if ( !group.isEmpty() ) + ingredients_html += "<ul>"; + } + + TQString amount_str = MixedNumber( ( *ing_it ).amount ).toString( number_format ); + + if ( (*ing_it).amount_offset > 0 ) + amount_str += "-"+MixedNumber( ( *ing_it ).amount + ( *ing_it ).amount_offset ).toString( number_format ); + else if ( ( *ing_it ).amount <= 1e-10 ) + amount_str = ""; + + TQString unit = ( *ing_it ).units.determineName( ( *ing_it ).amount + ( *ing_it ).amount_offset, useAbbreviations ); + + TQString tmp_format( ingredient_format ); + tmp_format.replace( TQRegExp( TQString::fromLatin1( "%n" ) ), TQStyleSheet::escape( ( *ing_it ).name ) ); + tmp_format.replace( TQRegExp( TQString::fromLatin1( "%a" ) ), amount_str ); + tmp_format.replace( TQRegExp( TQString::fromLatin1( "%u" ) ), TQStyleSheet::escape(unit) ); + tmp_format.replace( TQRegExp( TQString::fromLatin1( "%p" ) ), ( ( *ing_it ).prepMethodList.count() == 0 ) ? + TQString::fromLatin1( "" ) : TQString::fromLatin1( "; " ) + TQStyleSheet::escape( ( *ing_it ).prepMethodList.join(",") ) ); + + if ( (*ing_it).substitutes.count() > 0 ) + tmp_format += ", "+i18n("or"); + + ingredients_html += TQString( "<li>%1</li>" ).arg( tmp_format ); + + for ( TQValueList<IngredientData>::const_iterator sub_it = (*ing_it).substitutes.begin(); sub_it != (*ing_it).substitutes.end(); ) { + TQString amount_str = MixedNumber( ( *sub_it ).amount ).toString( number_format ); + + if ( (*ing_it).amount_offset > 0 ) + amount_str += "-"+MixedNumber( ( *sub_it ).amount + ( *sub_it ).amount_offset ).toString( number_format ); + else if ( ( *sub_it ).amount <= 1e-10 ) + amount_str = ""; + + TQString unit = ( *sub_it ).units.determineName( ( *sub_it ).amount + ( *sub_it ).amount_offset, config->readBoolEntry("AbbreviateUnits") ); + + TQString tmp_format( ingredient_format ); + tmp_format.replace( TQRegExp( TQString::fromLatin1( "%n" ) ), TQStyleSheet::escape( ( *sub_it ).name ) ); + tmp_format.replace( TQRegExp( TQString::fromLatin1( "%a" ) ), amount_str ); + tmp_format.replace( TQRegExp( TQString::fromLatin1( "%u" ) ), TQStyleSheet::escape(unit) ); + tmp_format.replace( TQRegExp( TQString::fromLatin1( "%p" ) ), ( ( *sub_it ).prepMethodList.count() == 0 ) ? + TQString::fromLatin1( "" ) : TQString::fromLatin1( "; " ) + TQStyleSheet::escape( ( *sub_it ).prepMethodList.join(",") ) ); + + ++sub_it; + if ( sub_it != (*ing_it).substitutes.end() ) + tmp_format += ", "+i18n("or"); + ingredients_html += TQString( "<li>%1</li>" ).arg( tmp_format ); + } + } + + if ( !group.isEmpty() ) + ingredients_html += "</ul>"; + } + if ( !ingredients_html.isEmpty() ) { + ingredients_html.prepend( "<table><tr><td valign=\"top\"><ul>" ); + ingredients_html.append( "</ul></td></tr></table>" ); + } + content = content.replace( "**INGREDIENTS**", HTMLIfVisible("ingredients",ingredients_html) ); + + //=======================PROPERTIES======================// + TQString properties_html; + + TQStringList hiddenList = config->readListEntry("HiddenProperties"); + IngredientPropertyList visibleProperties; + for ( IngredientPropertyList::const_iterator prop_it = recipe.properties.begin(); prop_it != recipe.properties.end(); ++prop_it ) { + if ( hiddenList.find((*prop_it).name) == hiddenList.end() ) + visibleProperties.append( *prop_it ); + } + + cols_it = m_columnsMap.find("properties"); + cols = 1; + if ( cols_it != m_columnsMap.end() ) + cols = cols_it.data(); + per_col = visibleProperties.count() / cols; + if ( visibleProperties.count() % cols != 0 ) //round up if division is not exact + per_col++; + + count = 0; + for ( IngredientPropertyList::const_iterator prop_it = visibleProperties.begin(); prop_it != visibleProperties.end(); ++prop_it ) { + if ( count != 0 && count % per_col == 0 ) + properties_html.append("</ul></td><td valign=\"top\"><ul>"); + + // if the amount given is <0, it means the property calculator found that the property was undefined for some ingredients, so the amount will be actually bigger + + TQString amount_str; + + double prop_amount = (*prop_it).amount; + if ( prop_amount > 0 ) { //TODO: make the precision configuratble + prop_amount = double( tqRound( prop_amount * 10.0 ) ) / 10.0; //not a "chemistry experiment" ;) Let's only have one decimal place + amount_str = beautify( TDEGlobal::locale() ->formatNumber( prop_amount, 5 ) ); + } + else + amount_str = "0"; + + properties_html += TQString( "<li>%1: <nobr>%2 %3</nobr></li>" ) + .arg( TQStyleSheet::escape( (*prop_it).name ) ) + .arg( amount_str ) + .arg( TQStyleSheet::escape( (*prop_it).units ) ); + + ++count; + } + + if ( !properties_html.isEmpty() ) { + properties_html.prepend( "<table><tr><td valign=\"top\"><ul>" ); + properties_html.append( "</ul></td></tr></table>" ); + } + content = content.replace( "**PROPERTIES**", HTMLIfVisible("properties",properties_html) ); + + //=======================RATINGS======================// + TQString ratings_html; + if ( recipe.ratingList.count() > 0 ) + ratings_html += TQString("<b>%1:</b>").arg(i18n("Ratings")); + + int rating_total = 0; + double rating_sum = 0; + for ( RatingList::const_iterator rating_it = recipe.ratingList.begin(); rating_it != recipe.ratingList.end(); ++rating_it ) { + ratings_html += "<hr />"; + + if ( !( *rating_it ).rater.isEmpty() ) + ratings_html += "<p><b>"+( *rating_it ).rater+"</b></p>"; + + if ( (*rating_it).ratingCriteriaList.count() > 0 ) + ratings_html += "<table>"; + for ( RatingCriteriaList::const_iterator rc_it = (*rating_it).ratingCriteriaList.begin(); rc_it != (*rating_it).ratingCriteriaList.end(); ++rc_it ) { + TQString image_url = fi.baseName() + "_photos/" + TQString::number((*rc_it).stars) + "-stars.png"; + image_url = KURL::encode_string( image_url ); + ratings_html += "<tr><td>"+(*rc_it).name+":</td><td><img src=\""+image_url+"\" /></td></tr>"; + if ( !TQFile::exists( fi.dirPath(true) + "/" + image_url ) ) { + TQPixmap starPixmap = Rating::starsPixmap((*rc_it).stars,true); + starPixmap.save( fi.dirPath(true) + "/" + image_url, "PNG" ); + } + + rating_total++; + rating_sum += (*rc_it).stars; + } + if ( (*rating_it).ratingCriteriaList.count() > 0 ) + ratings_html += "</table>"; + + if ( !( *rating_it ).comment.isEmpty() ) + ratings_html += "<p><i>"+( *rating_it ).comment+"</i></p>"; + } + content = content.replace( "**RATINGS**", HTMLIfVisible("ratings",ratings_html) ); + + TQString overall_html; + if ( rating_total > 0 ) { + double average = int(2*rating_sum/rating_total)/2; + overall_html += TQString("<b>%1:</b>").arg(i18n("Overall Rating")); + TQString image_url = fi.baseName() + "_photos/" + TQString::number(average) + "-stars.png"; + image_url = KURL::encode_string( image_url ); + overall_html += "<img src=\""+image_url+"\" />"; + if ( !TQFile::exists( fi.dirPath(true) + "/" + image_url ) ) { + TQPixmap starPixmap = Rating::starsPixmap(average,true); + starPixmap.save( fi.dirPath(true) + "/" + image_url, "PNG" ); + } + } + content = content.replace( "**OVERALL_RATING**", HTMLIfVisible("overall_rating",overall_html) ); +} + +void HTMLExporter::removeHTMLFiles( const TQString &filename, int recipe_id ) +{ + TQValueList<int> id; + id << recipe_id; + removeHTMLFiles( filename, id ); +} + +void HTMLExporter::removeHTMLFiles( const TQString &filename, const TQValueList<int> &recipe_ids ) +{ + //remove HTML file + TQFile old_file( filename + ".html" ); + if ( old_file.exists() ) + old_file.remove(); + + //remove photos + for ( TQValueList<int>::const_iterator it = recipe_ids.begin(); it != recipe_ids.end(); ++it ) { + TQFile photo( filename + "_photos/" + TQString::number(*it) + ".png" ); + if ( photo.exists() ) + photo.remove(); //remove photos in directory before removing it + } + + //take care of the default photo + TQFile photo( filename + "_photos/default_photo.png" ); + if ( photo.exists() ) photo.remove(); + + //remove photo directory + TQDir photo_dir; + photo_dir.rmdir( filename + "_photos" ); + + for ( double d = 0.5; d < 5.5; d += 0.5 ) { + if ( TQFile::exists(filename+"_photos/"+TQString::number(d)+"-stars.png") ) photo.remove(filename+"_photos/"+TQString::number(d)+"-stars.png"); + } +} + +TQString HTMLExporter::escape( const TQString & str ) +{ + TQString tmp( str ); + return tmp.replace( '/', "_" ); +} diff --git a/src/exporters/htmlexporter.h b/src/exporters/htmlexporter.h new file mode 100644 index 0000000..95cee4c --- /dev/null +++ b/src/exporters/htmlexporter.h @@ -0,0 +1,78 @@ +/*************************************************************************** +* Copyright (C) 2003-2006 by * +* Jason Kivlighn ([email protected] * +* * +* 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. * +***************************************************************************/ + +#ifndef HTMLEXPORTER_H +#define HTMLEXPORTER_H + +#include <tqdom.h> +#include <tqmap.h> + +#include "baseexporter.h" +#include "klomanager.h" + +class RecipeDB; +class KProgress; + +/** + * Exports a given recipe list as HTML + * @author Jason Kivlighn + */ +class HTMLExporter : public BaseExporter, protected KLOManager +{ +public: + HTMLExporter( const TQString&, const TQString& ); + virtual ~HTMLExporter(); + + virtual int supportedItems() const; + + static void removeHTMLFiles( const TQString &filename, int recipe_id ); + static void removeHTMLFiles( const TQString &filename, const TQValueList<int> &recipe_ids ); + + void setTemplate( const TQString &filename ); + void setStyle( const TQString &filename ); + +protected: + TQString createContent( const Recipe& recipe ); + virtual TQString createContent( const RecipeList & ); + virtual TQString createHeader( const RecipeList & ); + virtual TQString createFooter(); + + virtual int progressInterval() const { return 1; } + + virtual void loadBackgroundColor( const TQString &obj, const TQColor& ); + virtual void loadFont( const TQString &obj, const TQFont& ); + virtual void loadTextColor( const TQString &obj, const TQColor& ); + virtual void loadVisibility( const TQString &obj, bool ); + virtual void loadAlignment( const TQString &obj, int ); + virtual void loadBorder( const TQString &obj, const KreBorder& ); + virtual void loadColumns( const TQString & obj, int cols ); + + virtual void beginObject( const TQString &obj ); + virtual void endObject(); + + static TQString escape( const TQString & ); + + TQString m_templateContent; + +private: + void storePhoto( const Recipe &recipe ); + void populateTemplate( const Recipe &recipe, TQString &content ); + void replaceIfVisible( TQString &content, const TQString &name, const TQString &html ); + TQString HTMLIfVisible( const TQString &name, const TQString &html ); + + TQString m_layoutFilename; + TQString m_templateFilename; + TQString m_cachedCSS; + TQMap<TQString,bool> m_visibilityMap; + TQMap<TQString,int> m_columnsMap; + bool m_error; +}; + +#endif //HTMLEXPORTER_H diff --git a/src/exporters/kreexporter.cpp b/src/exporters/kreexporter.cpp new file mode 100644 index 0000000..e0d27c0 --- /dev/null +++ b/src/exporters/kreexporter.cpp @@ -0,0 +1,264 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Cyril Bosselut ([email protected]) * +* * +* Copyright (C) 2003-2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "kreexporter.h" + +#include <tqfile.h> +#include <tqstylesheet.h> +#include <tqbuffer.h> +#include <tqimage.h> + +#include <kdebug.h> +#include <tdelocale.h> +#include <kmdcodec.h> +#include <tdeglobal.h> +#include <kstandarddirs.h> + +#include "backends/recipedb.h" + +KreExporter::KreExporter( CategoryTree *_categories, const TQString& filename, const TQString &format ) : + BaseExporter( filename, format ), categories( _categories ) +{ + if ( format == "*.kre" ) { + setCompressed(true); + } +} + + +KreExporter::~KreExporter() +{ + delete categories; +} + +int KreExporter::supportedItems() const +{ + return RecipeDB::All; +} + +int KreExporter::headerFlags() const +{ + return RecipeDB::Categories; +} + +TQString KreExporter::createHeader( const RecipeList& recipes ) +{ + TQString xml; + + xml += "<?xml version=\"1.0\" encoding=\"UTF-8\" ?>\n"; + xml += "<krecipes version=\"" + krecipes_version() + "\" lang=\"" + ( TDEGlobal::locale() )->language() + "\" xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\" xsi:noNamespaceSchemaLocation=\"krecipes.xsd\">\n"; + + createCategoryStructure( xml, recipes ); + + return xml; +} + +TQString KreExporter::createFooter() +{ + return "</krecipes>\n"; +} + +TQString KreExporter::generateIngredient( const IngredientData &ing ) +{ + TQString xml; + + xml += "<name>" + TQStyleSheet::escape( ( ing ).name ) + "</name>\n"; + xml += "<amount>"; + if ( ing.amount_offset < 1e-10 ) { + xml += TQString::number( ing.amount ); + } + else { + xml += "<min>"+TQString::number( ing.amount )+"</min>"; + xml += "<max>"+TQString::number( ing.amount + ing.amount_offset )+"</max>"; + } + xml += "</amount>\n"; + TQString unit_str = ( ing.amount+ing.amount_offset > 1 ) ? ing.units.plural : ing.units.name; + xml += "<unit>" + TQStyleSheet::escape( unit_str ) + "</unit>\n"; + if ( ing.prepMethodList.count() > 0 ) + xml += "<prep>" + TQStyleSheet::escape( ing.prepMethodList.join(",") ) + "</prep>\n"; + + return xml; +} + +//TODO: use TQDOM (see recipemlexporter.cpp)? +TQString KreExporter::createContent( const RecipeList& recipes ) +{ + TQString xml; + + RecipeList::const_iterator recipe_it; + for ( recipe_it = recipes.begin(); recipe_it != recipes.end(); ++recipe_it ) { + + xml += "<krecipes-recipe>\n"; + xml += "<krecipes-description>\n"; + xml += "<title>" + TQStyleSheet::escape( ( *recipe_it ).title ) + "</title>\n"; + + for ( ElementList::const_iterator author_it = ( *recipe_it ).authorList.begin(); author_it != ( *recipe_it ).authorList.end(); ++author_it ) + xml += "<author>" + TQStyleSheet::escape( ( *author_it ).name ) + "</author>\n"; + + xml += "<pictures>\n"; + if ( !( *recipe_it ).photo.isNull() ) { + xml += "<pic format=\"JPEG\" id=\"1\"><![CDATA["; //fixed id until we implement multiple photos ability + TQByteArray data; + TQBuffer buffer( data ); + buffer.open( IO_WriteOnly ); + TQImageIO iio( &buffer, "JPEG" ); + iio.setImage( ( *recipe_it ).photo.convertToImage() ); + iio.write(); + + xml += KCodecs::base64Encode( data, true ); + + xml += "]]></pic>\n"; + } + xml += "</pictures>\n"; + xml += "<category>\n"; + + for ( ElementList::const_iterator cat_it = ( *recipe_it ).categoryList.begin(); cat_it != ( *recipe_it ).categoryList.end(); ++cat_it ) + xml += "<cat>" + TQStyleSheet::escape( ( *cat_it ).name ) + "</cat>\n"; + + xml += "</category>\n"; + xml += "<yield>"; + xml += "<amount>"; + if ( ( *recipe_it ).yield.amount_offset < 1e-10 ) { + xml += TQString::number( ( *recipe_it ).yield.amount ); + } + else { + xml += "<min>"+TQString::number( ( *recipe_it ).yield.amount )+"</min>"; + xml += "<max>"+TQString::number( ( *recipe_it ).yield.amount + ( *recipe_it ).yield.amount_offset )+"</max>"; + } + xml += "</amount>"; + xml += "<type>"+( *recipe_it ).yield.type+"</type>"; + xml += "</yield>\n"; + xml += "<preparation-time>"; + xml += ( *recipe_it ).prepTime.toString( "hh:mm" ); + xml += "</preparation-time>\n"; + xml += "</krecipes-description>\n"; + xml += "<krecipes-ingredients>\n"; + + IngredientList list_copy = ( *recipe_it ).ingList; + for ( IngredientList group_list = list_copy.firstGroup(); group_list.count() != 0; group_list = list_copy.nextGroup() ) { + TQString group = group_list[ 0 ].group; //just use the first's name... they're all the same + if ( !group.isEmpty() ) + xml += "<ingredient-group name=\"" + TQStyleSheet::escape(group) + "\">\n"; + + for ( IngredientList::const_iterator ing_it = group_list.begin(); ing_it != group_list.end(); ++ing_it ) { + xml += "<ingredient>\n"; + + xml += generateIngredient(*ing_it); + + if ( (*ing_it).substitutes.count() > 0 ) { + xml += "<substitutes>\n"; + for ( TQValueList<IngredientData>::const_iterator sub_it = (*ing_it).substitutes.begin(); sub_it != (*ing_it).substitutes.end(); ++sub_it ) { + xml += "<ingredient>\n"; + xml += generateIngredient(*sub_it); + xml += "</ingredient>\n"; + } + xml += "</substitutes>\n"; + } + + xml += "</ingredient>\n"; + } + + if ( !group.isEmpty() ) + xml += "</ingredient-group>\n"; + } + + /// @todo add ingredient properties + + xml += "</krecipes-ingredients>\n"; + xml += "<krecipes-instructions>\n"; + xml += TQStyleSheet::escape( ( *recipe_it ).instructions ); + xml += "</krecipes-instructions>\n"; + + //ratings + xml += "<krecipes-ratings>"; + for ( RatingList::const_iterator rating_it = (*recipe_it).ratingList.begin(); rating_it != (*recipe_it).ratingList.end(); ++rating_it ) { + xml += "<rating>"; + xml += "<comment>"+TQStyleSheet::escape( ( *rating_it ).comment )+"</comment>"; + xml += "<rater>"+TQStyleSheet::escape( ( *rating_it ).rater )+"</rater>"; + + xml += "<criterion>"; + for ( RatingCriteriaList::const_iterator rc_it = (*rating_it).ratingCriteriaList.begin(); rc_it != (*rating_it).ratingCriteriaList.end(); ++rc_it ) { + xml += "<criteria>"; + xml += "<name>"+(*rc_it).name+"</name>"; + xml += "<stars>"+TQString::number((*rc_it).stars)+"</stars>"; + xml += "</criteria>"; + } + xml += "</criterion>"; + xml += "</rating>"; + } + xml += "</krecipes-ratings>"; + + xml += "</krecipes-recipe>\n"; + } + + return xml; +} + +void KreExporter::createCategoryStructure( TQString &xml, const RecipeList &recipes ) +{ + TQValueList<int> categoriesUsed; + for ( RecipeList::const_iterator recipe_it = recipes.begin(); recipe_it != recipes.end(); ++recipe_it ) { + for ( ElementList::const_iterator cat_it = ( *recipe_it ).categoryList.begin(); cat_it != ( *recipe_it ).categoryList.end(); ++cat_it ) { + if ( categoriesUsed.find( ( *cat_it ).id ) == categoriesUsed.end() ) + categoriesUsed << ( *cat_it ).id; + } + } + + if ( !categoriesUsed.empty() ) { + //only keep the relevant category structure + removeIfUnused( categoriesUsed, categories ); + + xml += "<krecipes-category-structure>\n"; + writeCategoryStructure( xml, categories ); + xml += "</krecipes-category-structure>\n"; + } +} + +bool KreExporter::removeIfUnused( const TQValueList<int> &cat_ids, CategoryTree *parent, bool parent_should_show ) +{ + for ( CategoryTree * it = parent->firstChild(); it; it = it->nextSibling() ) { + if ( cat_ids.find( it->category.id ) != cat_ids.end() ) { + parent_should_show = true; + removeIfUnused( cat_ids, it, true ); //still recurse, but doesn't affect 'parent' + } + else { + bool result = removeIfUnused( cat_ids, it ); + if ( parent_should_show == false ) + parent_should_show = result; + } + } + + if ( !parent_should_show && parent->category.id != -1 ) { + //FIXME: CategoryTree is broken when deleting items + //delete parent; + + parent->category.id = -2; //temporary workaround + } + + return parent_should_show; +} + +void KreExporter::writeCategoryStructure( TQString &xml, const CategoryTree *categoryTree ) +{ + if ( categoryTree->category.id != -2 ) { + if ( categoryTree->category.id != -1 ) + xml += "<category name=\"" + TQStyleSheet::escape( categoryTree->category.name ).replace("\"",""") + "\">\n"; + + for ( CategoryTree * child_it = categoryTree->firstChild(); child_it; child_it = child_it->nextSibling() ) { + writeCategoryStructure( xml, child_it ); + } + + if ( categoryTree->category.id != -1 ) + xml += "</category>\n"; + } +} + diff --git a/src/exporters/kreexporter.h b/src/exporters/kreexporter.h new file mode 100644 index 0000000..d6b0b17 --- /dev/null +++ b/src/exporters/kreexporter.h @@ -0,0 +1,51 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Cyril Bosselut ([email protected]) * +* * +* Copyright (C) 2003-2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef KREEXPORTER_H +#define KREEXPORTER_H + +#include "baseexporter.h" +#include "datablocks/categorytree.h" + +class IngredientData; + +/** +Export class for Krecipes native file format (.kre, .kreml) + +@author Cyril Bosselut and Jason Kivlighn +*/ +class KreExporter : public BaseExporter +{ +public: + KreExporter( CategoryTree *, const TQString&, const TQString& ); + virtual ~KreExporter(); + + virtual int supportedItems() const; + +protected: + virtual TQString createContent( const RecipeList & ); + virtual TQString createHeader( const RecipeList & ); + virtual TQString createFooter(); + + virtual int headerFlags() const; + +private: + bool removeIfUnused( const TQValueList<int> &cat_ids, CategoryTree *parent, bool parent_should_show = false ); + void createCategoryStructure( TQString &xml, const RecipeList &recipes ); + void writeCategoryStructure( TQString &xml, const CategoryTree *categoryTree ); + TQString generateIngredient( const IngredientData &ing ); + + CategoryTree *categories; +}; + +#endif diff --git a/src/exporters/mmfexporter.cpp b/src/exporters/mmfexporter.cpp new file mode 100644 index 0000000..33ee1ec --- /dev/null +++ b/src/exporters/mmfexporter.cpp @@ -0,0 +1,220 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "mmfexporter.h" + +#include <tqregexp.h> + +#include <tdeconfig.h> +#include <kdebug.h> +#include <tdelocale.h> +#include <tdeapplication.h> + +#include "backends/recipedb.h" +#include "datablocks/mixednumber.h" +#include "mmdata.h" + +MMFExporter::MMFExporter( const TQString& filename, const TQString& format ) : + BaseExporter( filename, format ) +{} + + +MMFExporter::~MMFExporter() +{} + +int MMFExporter::supportedItems() const +{ + return RecipeDB::All ^ RecipeDB::Photo ^ RecipeDB::Ratings ^ RecipeDB::Authors; +} + +TQString MMFExporter::createContent( const RecipeList& recipes ) +{ + TQString content; + + RecipeList::const_iterator recipe_it; + for ( recipe_it = recipes.begin(); recipe_it != recipes.end(); ++recipe_it ) { + writeMMFHeader( content, *recipe_it ); + content += "\n"; + writeMMFIngredients( content, *recipe_it ); + content += "\n"; + writeMMFDirections( content, *recipe_it ); + content += "\n"; + + content += "-----\n"; //end of recipe indicator + } + + return content; +} + +/* Header: + * Line 1 - contains five hyphens and "Meal-Master" somewhere in the line + * Line 2 - "Title:" followed by a blank space; maximum of 60 char + * Line 3 - "Categories:" followed by a blank space; Maximum of 5 + * Line 4 - Numeric quantity representing the # of servings (1-9999) + */ +void MMFExporter::writeMMFHeader( TQString &content, const Recipe &recipe ) +{ + content += TQString( "----- Exported by Krecipes v%1 [Meal-Master Export Format] -----\n\n" ).arg( krecipes_version() ); + + TQString title = recipe.title; + title.truncate( 60 ); + content += " Title: " + title + "\n"; + + int i = 0; + TQStringList categories; + for ( ElementList::const_iterator cat_it = recipe.categoryList.begin(); cat_it != recipe.categoryList.end(); ++cat_it ) { + i++; + + if ( i == 6 ) + break; //maximum of 5 categories + categories << ( *cat_it ).name; + } + TQString cat_str = " Categories: " + categories.join( ", " ); + cat_str.truncate( 67 ); + content += cat_str + "\n"; + + content += " Servings: " + TQString::number( TQMIN( 9999, recipe.yield.amount ) ) + "\n"; //some yield info is lost here, but hey, that's the MM format +} + +/* Ingredient lines: + * Positions 1-7 contains a numeric quantity + * Positions 9-10 Meal-Master unit of measure codes + * Positions 12-39 contain text for an ingredient name, or a "-" + * in position 12 and text in positions 13-39 (the latter is a + * "continuation" line for the previous ingredient name) + */ +void MMFExporter::writeMMFIngredients( TQString &content, const Recipe &recipe ) +{ + //this format requires ingredients without a group to be written first + for ( IngredientList::const_iterator ing_it = recipe.ingList.begin(); ing_it != recipe.ingList.end(); ++ing_it ) { + if ( ( *ing_it ).groupID == -1 ) { + writeSingleIngredient( content, *ing_it, (*ing_it).substitutes.count() > 0 ); + + for ( TQValueList<IngredientData>::const_iterator sub_it = (*ing_it).substitutes.begin(); sub_it != (*ing_it).substitutes.end(); ) { + TQValueList<IngredientData>::const_iterator save_it = sub_it; + + ++sub_it; + writeSingleIngredient( content, *save_it, sub_it != (*ing_it).substitutes.end() ); + } + } + } + + IngredientList list_copy = recipe.ingList; + for ( IngredientList group_list = list_copy.firstGroup(); group_list.count() != 0; group_list = list_copy.nextGroup() ) { + if ( group_list[ 0 ].groupID == -1 ) //we already handled this group + continue; + + TQString group = group_list[ 0 ].group.left( 76 ); //just use the first's name... they're all the same + if ( !group.isEmpty() ) { + int length = group.length(); + TQString filler_lt = TQString().fill( '-', ( 76 - length ) / 2 ); + TQString filler_rt = ( length % 2 ) ? TQString().fill( '-', ( 76 - length ) / 2 + 1 ) : filler_lt; + content += filler_lt + group + filler_rt + "\n"; + } + + for ( IngredientList::const_iterator ing_it = group_list.begin(); ing_it != group_list.end(); ++ing_it ) { + writeSingleIngredient( content, *ing_it, (*ing_it).substitutes.count() > 0 ); + + for ( TQValueList<IngredientData>::const_iterator sub_it = (*ing_it).substitutes.begin(); sub_it != (*ing_it).substitutes.end(); ) { + TQValueList<IngredientData>::const_iterator save_it = sub_it; + + ++sub_it; + writeSingleIngredient( content, *save_it, sub_it != (*ing_it).substitutes.end() ); + } + } + } +} + +void MMFExporter::writeSingleIngredient( TQString &content, const Ingredient &ing, bool is_sub ) +{ + TDEConfig * config = kapp->config(); + config->setGroup( "Formatting" ); + MixedNumber::Format number_format = ( config->readBoolEntry( "Fraction" ) ) ? MixedNumber::MixedNumberFormat : MixedNumber::DecimalFormat; + + //columns 1-7 + if ( ing.amount > 0 ) + content += MixedNumber( ing.amount ).toString( number_format, false ).rightJustify( 7, ' ', true ) + " "; + else + content += " "; + + //columns 9-10 + bool found_short_form = false; + for ( int i = 0; unit_info[ i ].short_form; i++ ) { + if ( unit_info[ i ].expanded_form == ing.units.name || + unit_info[ i ].plural_expanded_form == ing.units.plural || + unit_info[ i ].short_form == ing.units.name ) { + found_short_form = true; + content += TQString( unit_info[ i ].short_form ).leftJustify( 2 ) + " "; + break; + } + } + if ( !found_short_form ) { + kdDebug() << "Warning: unable to find Meal-Master abbreviation for: " << ing.units.name << endl; + kdDebug() << " This ingredient (" << ing.name << ") will be exported without a unit" << endl; + content += " "; + } + + //columns 12-39 + TQString ing_name( ing.name ); + if ( ing.prepMethodList.count() > 0 ) + ing_name += "; " + ing.prepMethodList.join(", "); + + if ( is_sub ) + ing_name += ", or"; + + if ( !found_short_form ) + ing_name.prepend( ( ing.amount > 1 ? ing.units.plural : ing.units.name ) + " " ); + + //try and split the ingredient on a word boundry + int split_index; + if ( ing_name.length() > 28 ) { + split_index = ing_name.left(28).findRev(" ")+1; + if ( split_index == 0 ) + split_index = 28; + } + else + split_index = 28; + + content += ing_name.left(split_index) + "\n"; + + for ( unsigned int i = 0; i < ( ing_name.length() - 1 ) / 28; i++ ) //if longer than 28 chars, continue on next line(s) + content += " -" + ing_name.mid( 28 * ( i ) + split_index, 28 ) + "\n"; +} + +void MMFExporter::writeMMFDirections( TQString &content, const Recipe &recipe ) +{ + TQStringList lines = TQStringList::split("\n",recipe.instructions,true); + for ( TQStringList::const_iterator it = lines.begin(); it != lines.end(); ++it ) { + content += wrapText( *it, 80 ).join( "\n" ) + "\n"; + } +} + +TQStringList MMFExporter::wrapText( const TQString& str, int at ) const +{ + TQStringList ret; + TQString copy( str ); + bool stop = false; + while ( !stop ) { + TQString line( copy.left( at ) ); + if ( line.length() >= copy.length() ) + stop = true; + else { + TQRegExp rxp( "(\\s\\S*)$", false ); // last word in the new line + rxp.setMinimal( true ); // one word, only one word, please + line = line.replace( rxp, "" ); // remove last word + } + copy = copy.remove( 0, line.length() ); + line = line.stripWhiteSpace(); + line.prepend( " " ); // indent line by one char + ret << line; // output of current line + } + + return ret; +} diff --git a/src/exporters/mmfexporter.h b/src/exporters/mmfexporter.h new file mode 100644 index 0000000..6cb1d8a --- /dev/null +++ b/src/exporters/mmfexporter.h @@ -0,0 +1,51 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef MMFEXPORTER_H +#define MMFEXPORTER_H + +#include "baseexporter.h" + +/** + * Export class for the Meal-Master file format + * @author Jason Kivlighn + * + * Note: This format does not handle all the properties of recipes. + * Data lost in export to this format include: + * -Recipe photo + * -Authors + * -5 category maximum + * -Title is limited to 60 characters + * -Servings are limited to the range of 0-9999 + * -Units are limited: If a given unit does not have a + * corresponding MM abbrev., otherwise it will be + * exported without a unit. + */ +class MMFExporter : public BaseExporter +{ +public: + MMFExporter( const TQString&, const TQString& ); + virtual ~MMFExporter(); + + virtual int supportedItems() const; + +protected: + virtual TQString createContent( const RecipeList & ); + +private: + void writeMMFHeader( TQString &content, const Recipe &recipe ); + void writeMMFIngredients( TQString &content, const Recipe &recipe ); + void writeSingleIngredient( TQString &content, const Ingredient &ing, bool is_sub = false ); + void writeMMFDirections( TQString &content, const Recipe &recipe ); + + TQStringList wrapText( const TQString& str, int at ) const; +}; + +#endif //MMFEXPORTER_H diff --git a/src/exporters/plaintextexporter.cpp b/src/exporters/plaintextexporter.cpp new file mode 100644 index 0000000..db9902e --- /dev/null +++ b/src/exporters/plaintextexporter.cpp @@ -0,0 +1,176 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "plaintextexporter.h" + +#include <tdeconfig.h> +#include <tdeglobal.h> +#include <tdelocale.h> + +#include "backends/recipedb.h" + +PlainTextExporter::PlainTextExporter( const TQString& filename, const TQString& format ) : + BaseExporter( filename, format ) +{} + + +PlainTextExporter::~PlainTextExporter() +{} + +int PlainTextExporter::supportedItems() const +{ + return RecipeDB::All ^ RecipeDB::Photo; +} + +TQString PlainTextExporter::generateIngredient( const IngredientData &ing, MixedNumber::Format number_format ) +{ + TDEConfig *config = TDEGlobal::config(); + + TQString content; + + TQString amount_str = MixedNumber( ing.amount ).toString( number_format ); + + if ( ing.amount_offset > 0 ) + amount_str += "-"+MixedNumber( ing.amount + ing.amount_offset ).toString( number_format ); + else if ( ing.amount <= 1e-10 ) + amount_str = ""; + + content += amount_str; + if ( !amount_str.isEmpty() ) + content += " "; + + TQString unit_str = ing.units.determineName( ing.amount + ing.amount_offset, config->readBoolEntry("AbbreviateUnits") ); + + content += unit_str; + if ( !unit_str.isEmpty() ) + content += " "; + + content += ing.name; + + if ( ing.prepMethodList.count() > 0 ) + content += "; "+ing.prepMethodList.join(", "); + + return content; +} + + +TQString PlainTextExporter::createContent( const RecipeList& recipes ) +{ + TDEConfig *config = TDEGlobal::config(); + config->setGroup( "Formatting" ); + + MixedNumber::Format number_format = ( config->readBoolEntry( "Fraction" ) ) ? MixedNumber::MixedNumberFormat : MixedNumber::DecimalFormat; + + TQString content; + + RecipeList::const_iterator recipe_it; + for ( recipe_it = recipes.begin(); recipe_it != recipes.end(); ++recipe_it ) { + content += ( *recipe_it ).title + "\n\n"; + + if ( ( *recipe_it ).authorList.count() > 0 ) { + content += TQString("%1: ").arg(i18n("Authors")); + content += ( *recipe_it ).authorList.join(", "); + content += "\n"; + } + + if ( ( *recipe_it ).categoryList.count() > 0 ) { + content += TQString("%1: ").arg(i18n("Categories")); + content += ( *recipe_it ).categoryList.join(", "); + content += "\n"; + } + + if ( ( *recipe_it ).yield.amount > 0 ) { + content += TQString("%1: ").arg(i18n("Yields")); + content += ( *recipe_it ).yield.toString(); + content += "\n"; + } + + if ( !( *recipe_it ).prepTime.isNull() ) { + content += TQString("%1: ").arg(i18n("Preparation Time")); + content += ( *recipe_it ).prepTime.toString( "hh:mm" ); + content += "\n"; + } + + content += "\n"; + + IngredientList list_copy = ( *recipe_it ).ingList; + for ( IngredientList group_list = list_copy.firstGroup(); group_list.count() != 0; group_list = list_copy.nextGroup() ) { + TQString group = group_list[ 0 ].group; //just use the first's name... they're all the same + if ( !group.isEmpty() ) + content += group + ":\n"; + + for ( IngredientList::const_iterator ing_it = group_list.begin(); ing_it != group_list.end(); ++ing_it ) { + if ( !group.isEmpty() ) + content += " "; + + content += generateIngredient(*ing_it,number_format); + + if ( (*ing_it).substitutes.count() > 0 ) + content += ", "+i18n("or"); + content += "\n"; + + for ( TQValueList<IngredientData>::const_iterator sub_it = (*ing_it).substitutes.begin(); sub_it != (*ing_it).substitutes.end(); ) { + if ( !group.isEmpty() ) + content += " "; + + content += generateIngredient(*sub_it,number_format); + sub_it++; + if ( sub_it != (*ing_it).substitutes.end() ) + content += ", "+i18n("or"); + content += "\n"; + } + } + } + + content += "\n"; + + /// @todo add ingredient properties + + content += ( *recipe_it ).instructions; + + content += "\n\n"; + + if ( (*recipe_it).ratingList.count() > 0 ) + content += "----------"+i18n("Ratings")+"----------\n"; + + for ( RatingList::const_iterator rating_it = (*recipe_it).ratingList.begin(); rating_it != (*recipe_it).ratingList.end(); ++rating_it ) { + if ( !( *rating_it ).rater.isEmpty() ) + content += " "+( *rating_it ).rater+"\n"; + + if ( (*rating_it).ratingCriteriaList.size() > 0 ) + content += "\n"; + + for ( RatingCriteriaList::const_iterator rc_it = (*rating_it).ratingCriteriaList.begin(); rc_it != (*rating_it).ratingCriteriaList.end(); ++rc_it ) { + //FIXME: This is an ugly hack, but I don't know how else to be i18n friendly (if this is even that) + // and still be able to display the amount as a fraction + TQString starsTrans = i18n("1 star","%n stars",tqRound((*rc_it).stars)); + starsTrans.replace(TQString::number(tqRound((*rc_it).stars)),MixedNumber((*rc_it).stars).toString()); + + content += " "+(*rc_it).name+": "+starsTrans+"\n"; + } + + if ( (*rating_it).ratingCriteriaList.size() > 0 ) + content += "\n"; + + if ( !( *rating_it ).comment.isEmpty() ) + content += " "+( *rating_it ).comment+"\n"; + + content += "\n"; + } + + if ( (*recipe_it).ratingList.size() > 0 ) + content += "\n"; + + content += "-----\n\n"; //end of recipe indicator + } + + return content; +} + diff --git a/src/exporters/plaintextexporter.h b/src/exporters/plaintextexporter.h new file mode 100644 index 0000000..1f4a727 --- /dev/null +++ b/src/exporters/plaintextexporter.h @@ -0,0 +1,42 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef PLAINTEXTEXPORTER_H +#define PLAINTEXTEXPORTER_H + +#include "baseexporter.h" + +#include "datablocks/mixednumber.h" + +class IngredientData; + +/** + * Export class to export recipes into text format. + * Recipes exported with this class are not meant to be imported back + * into Krecipes or any other program. + * + * @author Jason Kivlighn + */ +class PlainTextExporter : public BaseExporter +{ +public: + PlainTextExporter( const TQString&, const TQString& ); + virtual ~PlainTextExporter(); + + virtual int supportedItems() const; + +protected: + virtual TQString createContent( const RecipeList & ); + +private: + TQString generateIngredient( const IngredientData &ing, MixedNumber::Format number_format ); +}; + +#endif //PLAINTEXTEXPORTER_H diff --git a/src/exporters/recipemlexporter.cpp b/src/exporters/recipemlexporter.cpp new file mode 100644 index 0000000..4831004 --- /dev/null +++ b/src/exporters/recipemlexporter.cpp @@ -0,0 +1,195 @@ +/*************************************************************************** +* Copyright (C) 2003-2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "recipemlexporter.h" + +#include <kdebug.h> +#include <tdelocale.h> + +#include "backends/recipedb.h" + +RecipeMLExporter::RecipeMLExporter( const TQString& filename, const TQString& format ) : + BaseExporter( filename, format ) +{} + + +RecipeMLExporter::~RecipeMLExporter() +{} + +int RecipeMLExporter::supportedItems() const +{ + return RecipeDB::All ^ RecipeDB::Photo ^ RecipeDB::Ratings; +} + +TQString RecipeMLExporter::createHeader( const RecipeList& ) +{ + TQString xml = "<?xml version=\"1.0\" encoding=\"UTF-8\" ?>\n"; + xml += "<!DOCTYPE recipeml PUBLIC \"-//FormatData//DTD RecipeML 0.5//EN\" \ + \"http://www.formatdata.com/recipeml/recipeml.dtd\">"; + xml += "<recipeml version=\"0.5\" generator=\"Krecipes v"+krecipes_version()+"\">\n"; + return xml; +} + +TQString RecipeMLExporter::createFooter() +{ + return "</recipeml>"; +} + +void RecipeMLExporter::createIngredient( TQDomElement &ing_tag, const IngredientData &ing, TQDomDocument &doc ) +{ + TQDomElement amt_tag = doc.createElement( "amt" ); + ing_tag.appendChild( amt_tag ); + + TQDomElement qty_tag = doc.createElement( "qty" ); + amt_tag.appendChild( qty_tag ); + if ( ing.amount_offset < 1e-10 ) + qty_tag.appendChild( doc.createTextNode( TQString::number( ing.amount ) ) ); + else { + TQDomElement range_tag = doc.createElement( "range" ); + qty_tag.appendChild(range_tag); + + TQDomElement q1_tag = doc.createElement( "q1" ); + q1_tag.appendChild( doc.createTextNode( TQString::number( ing.amount ) ) ); + TQDomElement q2_tag = doc.createElement( "q2" ); + q2_tag.appendChild( doc.createTextNode( TQString::number( ing.amount + ing.amount_offset ) ) ); + + range_tag.appendChild(q1_tag); + range_tag.appendChild(q2_tag); + } + + TQDomElement unit_tag = doc.createElement( "unit" ); + amt_tag.appendChild( unit_tag ); + unit_tag.appendChild( doc.createTextNode( ( ing.amount > 1 ) ? ing.units.plural : ing.units.name ) ); + + TQDomElement item_tag = doc.createElement( "item" ); + item_tag.appendChild( doc.createTextNode( ing.name ) ); + ing_tag.appendChild( item_tag ); + + TQDomElement prep_tag = doc.createElement( "prep" ); + prep_tag.appendChild( doc.createTextNode( ing.prepMethodList.join(",") ) ); + ing_tag.appendChild( prep_tag ); +} + +TQString RecipeMLExporter::createContent( const RecipeList& recipes ) +{ + TQDomDocument doc; + + RecipeList::const_iterator recipe_it; + for ( recipe_it = recipes.begin(); recipe_it != recipes.end(); ++recipe_it ) { + TQDomElement recipe_tag = doc.createElement( "recipe" ); + + doc.appendChild(recipe_tag); + //recipe_root.appendChild( recipe_tag ); //will append to either <menu> if exists or else <recipeml> + + TQDomElement head_tag = doc.createElement( "head" ); + recipe_tag.appendChild( head_tag ); + + TQDomElement title_tag = doc.createElement( "title" ); + title_tag.appendChild( doc.createTextNode( ( *recipe_it ).title ) ); + head_tag.appendChild( title_tag ); + + TQDomElement source_tag = doc.createElement( "source" ); + for ( ElementList::const_iterator author_it = ( *recipe_it ).authorList.begin(); author_it != ( *recipe_it ).authorList.end(); ++author_it ) { + TQDomElement srcitem_tag = doc.createElement( "srcitem" ); + srcitem_tag.appendChild( doc.createTextNode( ( *author_it ).name ) ); + source_tag.appendChild( srcitem_tag ); + } + head_tag.appendChild( source_tag ); + + TQDomElement categories_tag = doc.createElement( "categories" ); + for ( ElementList::const_iterator cat_it = ( *recipe_it ).categoryList.begin(); cat_it != ( *recipe_it ).categoryList.end(); ++cat_it ) { + TQDomElement cat_tag = doc.createElement( "cat" ); + cat_tag.appendChild( doc.createTextNode( ( *cat_it ).name ) ); + categories_tag.appendChild( cat_tag ); + } + head_tag.appendChild( categories_tag ); + + TQDomElement yield_tag = doc.createElement( "yield" ); + if ( ( *recipe_it ).yield.amount_offset < 1e-10 ) + yield_tag.appendChild( doc.createTextNode( TQString::number( ( *recipe_it ).yield.amount ) ) ); + else { + TQDomElement range_tag = doc.createElement( "range" ); + yield_tag.appendChild(range_tag); + + TQDomElement q1_tag = doc.createElement( "q1" ); + q1_tag.appendChild( doc.createTextNode( TQString::number(( *recipe_it ).yield.amount ) ) ); + TQDomElement q2_tag = doc.createElement( "q2" ); + q2_tag.appendChild( doc.createTextNode( TQString::number( ( *recipe_it ).yield.amount + ( *recipe_it ).yield.amount_offset ) ) ); + + range_tag.appendChild(q1_tag); + range_tag.appendChild(q2_tag); + } + if ( !( *recipe_it ).yield.type.isEmpty() ) { + TQDomElement yield_unit_tag = doc.createElement( "unit" ); + yield_unit_tag.appendChild( doc.createTextNode(( *recipe_it ).yield.type) ); + yield_tag.appendChild( yield_unit_tag ); + } + + head_tag.appendChild( yield_tag ); + + if ( !( *recipe_it ).prepTime.isNull() ) { + TQDomElement preptime_tag = doc.createElement( "preptime" ); + head_tag.appendChild( preptime_tag ); + preptime_tag.setAttribute( "type", i18n( "Total" ) ); + + TQDomElement preptime_time_tag = doc.createElement( "time" ); + preptime_tag.appendChild( preptime_time_tag ); + + TQDomElement preptime_min_qty_tag = doc.createElement( "qty" ); + preptime_time_tag.appendChild( preptime_min_qty_tag ); + preptime_min_qty_tag.appendChild( doc.createTextNode( TQString::number( ( *recipe_it ).prepTime.minute() + ( *recipe_it ).prepTime.hour() * 60 ) ) ); + + TQDomElement preptime_min_unit_tag = doc.createElement( "timeunit" ); + preptime_time_tag.appendChild( preptime_min_unit_tag ); + preptime_min_unit_tag.appendChild( doc.createTextNode( "minutes" ) ); + } + + TQDomElement ingredients_tag = doc.createElement( "ingredients" ); + IngredientList list_copy = ( *recipe_it ).ingList; + for ( IngredientList group_list = list_copy.firstGroup(); group_list.count() != 0; group_list = list_copy.nextGroup() ) { + TQDomElement ing_root; + + TQString group = group_list[ 0 ].group; //just use the first's name... they're all the same + if ( !group.isEmpty() ) { + TQDomElement ingdiv_tag = doc.createElement( "ing-div" ); + TQDomElement title_tag = doc.createElement( "title" ); + title_tag.appendChild( doc.createTextNode( group ) ); + ingdiv_tag.appendChild( title_tag ); + ingredients_tag.appendChild( ingdiv_tag ); + ing_root = ingdiv_tag; + } + else + ing_root = ingredients_tag; + + for ( IngredientList::const_iterator ing_it = group_list.begin(); ing_it != group_list.end(); ++ing_it ) { + TQDomElement ing_tag = doc.createElement( "ing" ); + ing_root.appendChild( ing_tag ); + + createIngredient( ing_tag, *ing_it, doc ); + + for ( TQValueList<IngredientData>::const_iterator sub_it = (*ing_it).substitutes.begin(); sub_it != (*ing_it).substitutes.end(); ++sub_it ) { + TQDomElement alt_ing_tag = doc.createElement( "alt-ing" ); + ing_tag.appendChild( alt_ing_tag ); + createIngredient( alt_ing_tag, *sub_it, doc ); + } + } + } + recipe_tag.appendChild( ingredients_tag ); + + TQDomElement directions_tag = doc.createElement( "directions" ); + recipe_tag.appendChild( directions_tag ); + + TQDomElement step_tag = doc.createElement( "step" ); //we've just got everything in one step + directions_tag.appendChild( step_tag ); + step_tag.appendChild( doc.createTextNode( ( *recipe_it ).instructions ) ); + } + + return doc.toString(); +} diff --git a/src/exporters/recipemlexporter.h b/src/exporters/recipemlexporter.h new file mode 100644 index 0000000..33b45d9 --- /dev/null +++ b/src/exporters/recipemlexporter.h @@ -0,0 +1,45 @@ +/*************************************************************************** +* Copyright (C) 2003-2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef RECIPEMLEXPORTER_H +#define RECIPEMLEXPORTER_H + +#include "baseexporter.h" + +#include <tqdom.h> + +class IngredientData; + +/** + * Export class for the RecipeML file format <http://www.formatdata.com/recipeml> + * @author Jason Kivlighn + * + * Note: This format does not handle all the properties of recipes. + * Data lost in export to this format include: + * -Recipe photo + */ +class RecipeMLExporter : public BaseExporter +{ +public: + RecipeMLExporter( const TQString&, const TQString& ); + virtual ~RecipeMLExporter(); + + virtual int supportedItems() const; + +protected: + virtual TQString createContent( const RecipeList& ); + virtual TQString createHeader( const RecipeList& ); + virtual TQString createFooter(); + +private: + void createIngredient( TQDomElement &ing_tag, const IngredientData &, TQDomDocument &doc ); +}; + +#endif //RECIPEMLEXPORTER_H diff --git a/src/exporters/rezkonvexporter.cpp b/src/exporters/rezkonvexporter.cpp new file mode 100644 index 0000000..b286b65 --- /dev/null +++ b/src/exporters/rezkonvexporter.cpp @@ -0,0 +1,322 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "rezkonvexporter.h" + +#include <tqregexp.h> + +#include <tdeconfig.h> +#include <kdebug.h> +#include <tdelocale.h> +#include <tdeapplication.h> + +#include "backends/recipedb.h" +#include "datablocks/mixednumber.h" + +struct translate_unit_info +{ + const char *english; + const char *english_plural; + const char *german; + const char *german_plural; +}; + +static translate_unit_info translate_units[] = { + {"g", "g", "Gramm", "Gramms" }, + {"ml", "ml", "ml", "ml"}, + {"l", "l", "Ltr.", "Ltr."}, + {"kg", "kg", "kg", "kg"}, + {"mg", "mg", "mg", "mg"}, + {"teaspoon", "teaspoons", "Teel.", "Teel."}, + {"t.", "t.", "Teel.", "Teel."}, + {"t", "t", "Teel.", "Teel."}, + {"tsp.", "tsp.", "Teel.", "Teel."}, + {"tsp", "tsp", "Teel.", "Teel."}, + {"tablespoon", "tablespoons", "Essl.", "Essl."}, + {"tbsp.", "tbsp.", "Essl.", "Essl."}, + {"tbsp", "tbsp", "Essl.", "Essl."}, + {"T", "T", "Essl.", "Essl."}, + {"T.", "T.", "Essl.", "Essl."}, + {"cup", "cups", "Tasse", "Tassen"}, + {"c.", "c.", "Tasse", "Tassen"}, + {"can", "cans", "Dose", "Dosen"}, + {"drop", "drops", "Tropfen", "Tropfen"}, + {"large", "large", "gro�", "gro�"}, + {"medium", "medium", "mittl.", "mittl."}, + {"small", "small", "klein.", "klein."}, + {"pinch", "pinches", "Prise", "Prisen"}, + {"package", "packages", "Pack.", "Pack."}, + {"bunch", "bunches", "Bund.", "Bunde"}, + {"stem", "stems", "Strange", "Strangen"}, + {"twig", "twigs", "Zweig", "Zweige"}, + {"tip of a knife", "tips of a knife", "Messersp.", "Messersp."}, + {"sheet", "sheets", "Blatt", "Bl�tter"}, + {"handful", "handfuls", "Handvoll", "Handvoll"}, + {"head", "heads", "Kopf", "K�pfe"}, + {"slice", "slices", "Scheibe", "Sheiben"}, + {"some", "some", "Einige", "Einige"}, + {"a little", "a little", "Etwas", "Etwas"}, + {"little can", "little cans", "D�schen", "D�schen"}, + {"glass", "glasses", "Glas", "Gl�ser"}, + {"piece", "pieces", "St�ck", "St�ck"}, + {"pot", "pots", "Topf", "Topf"}, + {"generous", "generous", "Reichlich", "Reichlich"}, + {"dash", "dashes", "Spritzer", "Spritzer"}, + {"clove", "cloves", "Zehe", "Zehen"}, + {"slice", "slices", "Platte", "Platten"}, + {"shot", "shots", "Schuss", "Schuss"}, + {"peduncle", "peduncles", "Stiel", "Stiele"}, + {"heaping teaspoon", "heaping teaspoons", "geh. TL", "geh. TL"}, + {"heaping tsp.", "heaping tsp.", "geh. TL", "geh. TL"}, + {"heaping tsp.", "heaping tsp.", "geh. TL", "geh. TL"}, + {"heaping tablespoon", "heaping tablespoon", "geh. EL", "geh. EL"}, + {"heaping tbsp.", "heaping tbsp.", "geh. EL", "geh. EL"}, +#if 0 + {"pound", "pounds", "", ""}, + {"lb.", "lbs.", "", ""}, + {"ounce", "ounces", "", ""}, + {"oz.", "oz.", "", ""}, +#endif + {"", "", "", ""}, + { 0, 0, 0, 0 } + }; + +RezkonvExporter::RezkonvExporter( const TQString& filename, const TQString& format ) : + BaseExporter( filename, format ) +{} + + +RezkonvExporter::~RezkonvExporter() +{} + +int RezkonvExporter::supportedItems() const +{ + return RecipeDB::All ^ RecipeDB::Photo ^ RecipeDB::Ratings; +} + +TQString RezkonvExporter::createContent( const RecipeList& recipes ) +{ + TQString content; + + RecipeList::const_iterator recipe_it; + for ( recipe_it = recipes.begin(); recipe_it != recipes.end(); ++recipe_it ) { + writeHeader( content, *recipe_it ); + content += "\n"; + writeIngredients( content, *recipe_it ); + content += "\n"; + writeDirections( content, *recipe_it ); + content += "\n"; + + content += "=====\n\n"; //end of recipe indicator + } + + return content; +} + +/* Header: + * Line 1 - contains five hyphens and "Meal-Master" somewhere in the line + * Line 2 - "Title:" followed by a blank space; maximum of 60 char + * Line 3 - "Categories:" followed by a blank space; Maximum of 5 + * Line 4 - Numeric quantity representing the # of servings (1-9999) + */ +void RezkonvExporter::writeHeader( TQString &content, const Recipe &recipe ) +{ + content += TQString( "===== Exported by Krecipes v%1 [REZKONV Export Format] =====\n\n" ).arg( krecipes_version() ); + + TQString title = recipe.title; + title.truncate( 60 ); + content += TQString("Titel: ").rightJustify( 13, ' ', true) + title + "\n"; + + int i = 0; + TQStringList categories; + for ( ElementList::const_iterator cat_it = recipe.categoryList.begin(); cat_it != recipe.categoryList.end(); ++cat_it ) { + i++; + + if ( i == 6 ) + break; //maximum of 5 categories + categories << ( *cat_it ).name; + } + TQString cat_str = TQString("Kategorien: ").rightJustify( 13, ' ', true) + categories.join( ", " ); + cat_str.truncate( 67 ); + content += cat_str + "\n"; + + content += TQString("Menge: ").rightJustify( 13, ' ', true) + recipe.yield.toString() + "\n"; +} + +/* Ingredient lines: + * Positions 1-7 contains a numeric quantity + * Positions 9-10 Meal-Master unit of measure codes + * Positions 12-39 contain text for an ingredient name, or a "-" + * in position 12 and text in positions 13-39 (the latter is a + * "continuation" line for the previous ingredient name) + */ +void RezkonvExporter::writeIngredients( TQString &content, const Recipe &recipe ) +{ + //this format requires ingredients without a group to be written first + for ( IngredientList::const_iterator ing_it = recipe.ingList.begin(); ing_it != recipe.ingList.end(); ++ing_it ) { + if ( ( *ing_it ).groupID == -1 ) { + writeSingleIngredient( content, *ing_it, (*ing_it).substitutes.count() > 0 ); + + for ( TQValueList<IngredientData>::const_iterator sub_it = (*ing_it).substitutes.begin(); sub_it != (*ing_it).substitutes.end(); ) { + TQValueList<IngredientData>::const_iterator save_it = sub_it; + + ++sub_it; + writeSingleIngredient( content, *save_it, sub_it != (*ing_it).substitutes.end() ); + } + } + } + + IngredientList list_copy = recipe.ingList; + for ( IngredientList group_list = list_copy.firstGroup(); group_list.count() != 0; group_list = list_copy.nextGroup() ) { + if ( group_list[ 0 ].groupID == -1 ) //we already handled this group + continue; + + TQString group = group_list[ 0 ].group.left( 76 ); //just use the first's name... they're all the same + if ( !group.isEmpty() ) { + int length = group.length(); + TQString filler_lt = TQString().fill( '=', ( 76 - length ) / 2 ); + TQString filler_rt = ( length % 2 ) ? TQString().fill( '=', ( 76 - length ) / 2 + 1 ) : filler_lt; + content += filler_lt + group + filler_rt + "\n"; + } + + for ( IngredientList::const_iterator ing_it = group_list.begin(); ing_it != group_list.end(); ++ing_it ) { + writeSingleIngredient( content, *ing_it, (*ing_it).substitutes.count() > 0 ); + + for ( TQValueList<IngredientData>::const_iterator sub_it = (*ing_it).substitutes.begin(); sub_it != (*ing_it).substitutes.end(); ) { + TQValueList<IngredientData>::const_iterator save_it = sub_it; + + ++sub_it; + writeSingleIngredient( content, *save_it, sub_it != (*ing_it).substitutes.end() ); + } + } + } + + TQString authorLines; + if ( recipe.authorList.count() > 0 ) { + content += "============================== TQUELLE ==============================\n"; + authorLines = " "+(*recipe.authorList.begin()).name+"\n"; + } + for ( ElementList::const_iterator author_it = ++recipe.authorList.begin(); author_it != recipe.authorList.end(); ++author_it ) { + authorLines += " -- "; + authorLines += (*author_it).name + "\n"; + } + if ( !authorLines.isEmpty() ) + authorLines += "\n"; + content += authorLines; +} + +void RezkonvExporter::writeSingleIngredient( TQString &content, const IngredientData &ing, bool is_sub ) +{ + TDEConfig * config = kapp->config(); + config->setGroup( "Formatting" ); + MixedNumber::Format number_format = ( config->readBoolEntry( "Fraction" ) ) ? MixedNumber::MixedNumberFormat : MixedNumber::DecimalFormat; + + //columns 1-7 + if ( ing.amount > 1e-8 ) { + TQString amount_str = MixedNumber( ing.amount ).toString( number_format, false ); + + if ( ing.amount_offset > 0 ) + amount_str += "-"+MixedNumber( ing.amount + ing.amount_offset ).toString( number_format, false ); + + if ( amount_str.length() > 7 ) { //too long, let's try the other formatting + MixedNumber::Format other_format = (number_format == MixedNumber::DecimalFormat) ? MixedNumber::MixedNumberFormat : MixedNumber::DecimalFormat; + + TQString new_amount_str = MixedNumber( ing.amount ).toString( other_format, false ); + + if ( ing.amount_offset > 0 ) + new_amount_str += "-"+MixedNumber( ing.amount + ing.amount_offset ).toString( other_format, false ); + + if (new_amount_str.length() > 7) { //still too long, use original formatting, but truncate it + amount_str = amount_str.left(7); + kdDebug()<<"Warning: Amount text too long, truncating"<<endl; + } + } + content += amount_str.rightJustify( 7, ' ', true ) + " "; + } + else + content += TQString().fill(' ',7+1); + + //columns 9-19 + bool found_translation = false; + for ( int i = 0; translate_units[ i ].english; i++ ) { + if ( translate_units[ i ].english == ing.units.name || translate_units[ i ].english_plural == ing.units.plural || translate_units[ i ].german == ing.units.name || translate_units[ i ].german_plural == ing.units.plural ) { + found_translation = true; + TQString unit; + if ( ing.amount > 1 ) + unit += translate_units[i].german; + else + unit += translate_units[i].german_plural; + content += unit.leftJustify( 9 ) + " "; + break; + } + } + if ( !found_translation ) { + kdDebug() << "Warning: unable to find German translation for: " << ing.units.name << endl; + kdDebug() << " This ingredient (" << ing.name << ") will be exported without a unit" << endl; + content += TQString().fill(' ',9+1); + } + + //columns 21-70 + TQString ing_name( ing.name ); + if ( ing.prepMethodList.count() > 0 ) + ing_name += "; " + ing.prepMethodList.join(", "); + + if ( is_sub ) + ing_name += ", or"; //FIXME: what's 'or' in German? + + if ( !found_translation ) + ing_name.prepend( ( ing.amount > 1 ? ing.units.plural : ing.units.name ) + " " ); + + //try and split the ingredient on a word boundry + int split_index; + if ( ing_name.length() > 50 ) { + split_index = ing_name.left(50).findRev(" ")+1; + if ( split_index == 0 ) + split_index = 50; + } + else + split_index = 50; + + content += ing_name.left(split_index) + "\n"; + + for ( unsigned int i = 0; i < ( ing_name.length() - 1 ) / 50; i++ ) //if longer than 50 chars, continue on next line(s) + content += TQString().fill(' ',(7+1)+(9+1)) + "-" + ing_name.mid( 50 * ( i ) + split_index, 50 ) + "\n"; +} + +void RezkonvExporter::writeDirections( TQString &content, const Recipe &recipe ) +{ + TQStringList lines = TQStringList::split("\n",recipe.instructions,true); + for ( TQStringList::const_iterator it = lines.begin(); it != lines.end(); ++it ) { + content += wrapText( *it, 80 ).join( "\n" ) + "\n"; + } +} + +TQStringList RezkonvExporter::wrapText( const TQString& str, int at ) const +{ + TQStringList ret; + TQString copy( str ); + bool stop = false; + while ( !stop ) { + TQString line( copy.left( at ) ); + if ( line.length() >= copy.length() ) + stop = true; + else { + TQRegExp rxp( "(\\s\\S*)$", false ); // last word in the new line + rxp.setMinimal( true ); // one word, only one word, please + line = line.replace( rxp, "" ); // remove last word + } + copy = copy.remove( 0, line.length() ); + line = line.stripWhiteSpace(); + line.prepend( " " ); // indent line by one char + ret << line; // output of current line + } + + return ret; +} diff --git a/src/exporters/rezkonvexporter.h b/src/exporters/rezkonvexporter.h new file mode 100644 index 0000000..32ffbd4 --- /dev/null +++ b/src/exporters/rezkonvexporter.h @@ -0,0 +1,51 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef REZKONVEXPORTER_H +#define REZKONVEXPORTER_H + +#include "baseexporter.h" + +/** + * Export class for the Rezkonv file format + * @author Jason Kivlighn + * + * Note: This format does not handle all the properties of recipes. + * Data lost in export to this format include: + * -Recipe photo + * -Authors + * -5 category maximum + * -Title is limited to 60 characters + * -Servings are limited to the range of 0-9999 + * -Units are limited: If a given unit does not have a + * corresponding MM abbrev., otherwise it will be + * exported without a unit. + */ +class RezkonvExporter : public BaseExporter +{ +public: + RezkonvExporter( const TQString&, const TQString& ); + virtual ~RezkonvExporter(); + + virtual int supportedItems() const; + +protected: + virtual TQString createContent( const RecipeList & ); + +private: + void writeHeader( TQString &content, const Recipe &recipe ); + void writeIngredients( TQString &content, const Recipe &recipe ); + void writeSingleIngredient( TQString &content, const IngredientData &ing, bool is_sub = false ); + void writeDirections( TQString &content, const Recipe &recipe ); + + TQStringList wrapText( const TQString& str, int at ) const; +}; + +#endif //REZKONVEXPORTER_H diff --git a/src/hi128-app-krecipes.png b/src/hi128-app-krecipes.png Binary files differnew file mode 100644 index 0000000..7d0200c --- /dev/null +++ b/src/hi128-app-krecipes.png diff --git a/src/hi16-app-krecipes.png b/src/hi16-app-krecipes.png Binary files differnew file mode 100644 index 0000000..13e1c76 --- /dev/null +++ b/src/hi16-app-krecipes.png diff --git a/src/hi22-app-krecipes.png b/src/hi22-app-krecipes.png Binary files differnew file mode 100644 index 0000000..3c8c354 --- /dev/null +++ b/src/hi22-app-krecipes.png diff --git a/src/hi32-app-krecipes.png b/src/hi32-app-krecipes.png Binary files differnew file mode 100644 index 0000000..5322c15 --- /dev/null +++ b/src/hi32-app-krecipes.png diff --git a/src/hi48-app-krecipes.png b/src/hi48-app-krecipes.png Binary files differnew file mode 100644 index 0000000..67b8b20 --- /dev/null +++ b/src/hi48-app-krecipes.png diff --git a/src/hi64-app-krecipes.png b/src/hi64-app-krecipes.png Binary files differnew file mode 100644 index 0000000..0616334 --- /dev/null +++ b/src/hi64-app-krecipes.png diff --git a/src/image.h b/src/image.h new file mode 100644 index 0000000..8558df8 --- /dev/null +++ b/src/image.h @@ -0,0 +1,1123 @@ +#ifndef IMAGE_H +#define IMAGE_H + +static const char * defaultPhoto[] = + { + "221 166 948 2", + " c None", + ". c #00001E", + "+ c #00001D", + "@ c #00001C", + "# c #00001B", + "$ c #00001A", + "% c #00001F", + "& c #000020", + "* c #000021", + "= c #000022", + "- c #1B1B25", + "; c #29292F", + "> c #353539", + ", c #37373B", + "' c #38383C", + ") c #414144", + "! c #404043", + "~ c #39393D", + "{ c #303035", + "] c #22222A", + "^ c #000023", + "/ c #353537", + "( c #4B4B4B", + "_ c #525252", + ": c #585858", + "< c #5B5B5B", + "[ c #5D5D5D", + "} c #5F5F5F", + "| c #616161", + "1 c #626262", + "2 c #636363", + "3 c #8B8B8B", + "4 c #DDDDDD", + "5 c #C7C7C8", + "6 c #A5A5A6", + "7 c #919193", + "8 c #7B7B7D", + "9 c #151522", + "0 c #444444", + "a c #4E4E4E", + "b c #555555", + "c c #5A5A5A", + "d c #5C5C5C", + "e c #5E5E5E", + "f c #646464", + "g c #676767", + "h c #999999", + "i c #E5E5E5", + "j c #DCDCDC", + "k c #D3D3D3", + "l c #CFCFCF", + "m c #CBCBCB", + "n c #C5C5C5", + "o c #969697", + "p c #707072", + "q c #000024", + "r c #26262C", + "s c #474747", + "t c #505050", + "u c #575757", + "v c #606060", + "w c #656565", + "x c #666666", + "y c #6D6D6D", + "z c #A7A7A7", + "A c #E7E7E7", + "B c #DADADA", + "C c #D4D4D4", + "D c #D1D1D1", + "E c #CCCCCC", + "F c #C6C6C6", + "G c #BFBFBF", + "H c #B7B7B7", + "I c #979798", + "J c #656568", + "K c #323235", + "L c #484848", + "M c #595959", + "N c #686868", + "O c #737373", + "P c #B6B6B6", + "Q c #E6E6E6", + "R c #D8D8D8", + "S c #D6D6D6", + "T c #D2D2D2", + "U c #CDCDCD", + "V c #C7C7C7", + "W c #C0C0C0", + "X c #AFAFAF", + "Y c #A5A5A5", + "Z c #868687", + "` c #33333A", + " . c #000025", + ".. c #39393B", + "+. c #4C4C4C", + "@. c #545454", + "#. c #696969", + "$. c #7C7C7C", + "%. c #C4C4C4", + "&. c #D9D9D9", + "*. c #D7D7D7", + "=. c #CECECE", + "-. c #C8C8C8", + ";. c #C1C1C1", + ">. c #B8B8B8", + ",. c #9A9A9A", + "'. c #8E8E8E", + "). c #424247", + "!. c #434343", + "~. c #4F4F4F", + "{. c #6A6A6A", + "]. c #6B6B6B", + "^. c #6C6C6C", + "/. c #878787", + "(. c #DBDBDB", + "_. c #C9C9C9", + ":. c #B0B0B0", + "<. c #8D8D8D", + "[. c #838383", + "}. c #3E3E43", + "|. c #000026", + "1. c #161624", + "2. c #464646", + "3. c #6E6E6E", + "4. c #DEDEDE", + "5. c #D5D5D5", + "6. c #D0D0D0", + "7. c #CACACA", + "8. c #B9B9B9", + "9. c #9B9B9B", + "0. c #69696A", + "a. c #27272D", + "b. c #4A4A4A", + "c. c #6F6F6F", + "d. c #707070", + "e. c #717171", + "f. c #E4E4E4", + "g. c #DFDFDF", + "h. c #C2C2C2", + "i. c #BABABA", + "j. c #8C8C8C", + "k. c #6F6F70", + "l. c #000027", + "m. c #727272", + "n. c #757575", + "o. c #AAAAAA", + "p. c #E0E0E0", + "q. c #B1B1B1", + "r. c #9C9C9C", + "s. c #828282", + "t. c #676768", + "u. c #3B3B3D", + "v. c #747474", + "w. c #E1E1E1", + "x. c #5E5E60", + "y. c #000028", + "z. c #3E3E40", + "A. c #535353", + "B. c #767676", + "C. c #E3E3E3", + "D. c #E2E2E2", + "E. c #8A8A8A", + "F. c #818181", + "G. c #5C5C5E", + "H. c #000029", + "I. c #161626", + "J. c #494949", + "K. c #565656", + "L. c #777777", + "M. c #787878", + "N. c #797979", + "O. c #989898", + "P. c #898989", + "Q. c #525255", + "R. c #181828", + "S. c #4D4D4D", + "T. c #7A7A7A", + "U. c #7B7B7B", + "V. c #979797", + "W. c #A4A4A4", + "X. c #47474C", + "Y. c #00002A", + "Z. c #292930", + "`. c #7D7D7D", + " + c #7E7E7E", + ".+ c #9E9E9E", + "++ c #AEAEAE", + "@+ c #A3A3A3", + "#+ c #969696", + "$+ c #7F7F7F", + "%+ c #393940", + "&+ c #38383B", + "*+ c #ACACAC", + "=+ c #ADADAD", + "-+ c #949494", + ";+ c #888888", + ">+ c #37373E", + ",+ c #00002B", + "'+ c #404042", + ")+ c #808080", + "!+ c #858585", + "~+ c #BBBBBB", + "{+ c #939393", + "]+ c #252532", + "^+ c #848484", + "/+ c #E8E8E8", + "(+ c #ABABAB", + "_+ c #A2A2A2", + ":+ c #919191", + "<+ c #868686", + "[+ c #00002C", + "}+ c #18182A", + "|+ c #E9E9E9", + "1+ c #A0A0A0", + "2+ c #8F8F8F", + "3+ c #727273", + "4+ c #2A2A32", + "5+ c #EBEBEB", + "6+ c #ECECEC", + "7+ c #B4B4B4", + "8+ c #A9A9A9", + "9+ c #00002D", + "0+ c #2F2F36", + "a+ c #A8A8A8", + "b+ c #EDEDED", + "c+ c #EFEFEF", + "d+ c #F0F0F0", + "e+ c #BEBEBE", + "f+ c #B3B3B3", + "g+ c #666668", + "h+ c #414143", + "i+ c #F4F4F4", + "j+ c #F5F5F5", + "k+ c #F1F1F1", + "l+ c #BDBDBD", + "m+ c #B2B2B2", + "n+ c #A6A6A6", + "o+ c #5D5D60", + "p+ c #00002E", + "q+ c #4A4A4B", + "r+ c #909090", + "s+ c #F9F9F9", + "t+ c #FCFCFC", + "u+ c #F6F6F6", + "v+ c #BCBCBC", + "w+ c #5A5A5D", + "x+ c #19192C", + "y+ c #929292", + "z+ c #EAEAEA", + "A+ c #F8F8F8", + "B+ c #FFFFFF", + "C+ c #4F4F53", + "D+ c #00002F", + "E+ c #232330", + "F+ c #959595", + "G+ c #F2F2F2", + "H+ c #FDFDFD", + "I+ c #44444A", + "J+ c #000030", + "K+ c #2F2F37", + "L+ c #EEEEEE", + "M+ c #9F9F9F", + "N+ c #373740", + "O+ c #3E3E42", + "P+ c #FAFAFA", + "Q+ c #9D9D9D", + "R+ c #34343E", + "S+ c #000031", + "T+ c #47474A", + "U+ c #232334", + "V+ c #A1A1A1", + "W+ c #000032", + "X+ c #1B1B2F", + "Y+ c #F3F3F3", + "Z+ c #303039", + "`+ c #666667", + " @ c #000033", + ".@ c #3A3A40", + "+@ c #B5B5B5", + "@@ c #5D5D5F", + "#@ c #48484B", + "$@ c #F7F7F7", + "%@ c #545458", + "&@ c #000034", + "*@ c #525254", + "=@ c #515155", + "-@ c #000035", + ";@ c #1B1B31", + ">@ c #46464C", + ",@ c #282837", + "'@ c #3B3B44", + ")@ c #000036", + "!@ c #35353F", + "~@ c #C3C3C3", + "{@ c #30303D", + "]@ c #45454A", + "^@ c #FBFBFB", + "/@ c #2C2C3A", + "(@ c #000037", + "_@ c #505053", + ":@ c #1D1D34", + "<@ c #04042B", + "[@ c #080822", + "}@ c #0B0B1F", + "|@ c #0F0F19", + "1@ c #121217", + "2@ c #11111A", + "3@ c #11111E", + "4@ c #12121F", + "5@ c #13131F", + "6@ c #0F0F25", + "7@ c #1C1C34", + "8@ c #02022C", + "9@ c #040422", + "0@ c #050522", + "a@ c #090916", + "b@ c #0C0C0C", + "c@ c #0F0F0F", + "d@ c #101010", + "e@ c #131313", + "f@ c #141414", + "g@ c #171717", + "h@ c #181818", + "i@ c #191919", + "j@ c #1A1A1A", + "k@ c #1C1C1C", + "l@ c #1D1D1D", + "m@ c #1C1C20", + "n@ c #191924", + "o@ c #0B0B2D", + "p@ c #000038", + "q@ c #1E1E36", + "r@ c #060623", + "s@ c #0A0A0A", + "t@ c #0B0B0B", + "u@ c #0E0E0E", + "v@ c #1F1F1F", + "w@ c #1E1E1E", + "x@ c #202020", + "y@ c #222222", + "z@ c #212121", + "A@ c #1B1B26", + "B@ c #151529", + "C@ c #353540", + "D@ c #555557", + "E@ c #06062D", + "F@ c #0B0B24", + "G@ c #0E0E19", + "H@ c #121212", + "I@ c #161616", + "J@ c #151515", + "K@ c #242424", + "L@ c #252525", + "M@ c #232323", + "N@ c #282828", + "O@ c #262626", + "P@ c #212127", + "Q@ c #000039", + "R@ c #404047", + "S@ c #515153", + "T@ c #08082E", + "U@ c #11111F", + "V@ c #1B1B1B", + "W@ c #2C2C2C", + "X@ c #272727", + "Y@ c #2E2E2E", + "Z@ c #2B2B2B", + "`@ c #2A2A2A", + " # c #292929", + ".# c #232329", + "+# c #14142D", + "@# c #00003A", + "## c #515154", + "$# c #50505A", + "%# c #111127", + "&# c #19191D", + "*# c #333333", + "=# c #363636", + "-# c #2D2D2D", + ";# c #323232", + "># c #2F2F2F", + ",# c #15152E", + "'# c #000000", + ")# c #4D4D58", + "!# c #171727", + "~# c #353535", + "{# c #393939", + "]# c #404040", + "^# c #3F3F3F", + "/# c #313131", + "(# c #303030", + "_# c #373737", + ":# c #383838", + "<# c #16162F", + "[# c #00003B", + "}# c #3C3C3C", + "|# c #3B3B3B", + "1# c #010101", + "2# c #020202", + "3# c #3F3F4C", + "4# c #1B1B2A", + "5# c #343434", + "6# c #3D3D3D", + "7# c #454545", + "8# c #515151", + "9# c #414141", + "0# c #3E3E3E", + "a# c #171730", + "b# c #030303", + "c# c #040404", + "d# c #050505", + "e# c #7A7A7B", + "f# c #1F1F2D", + "g# c #424242", + "h# c #3A3A3A", + "i# c #2D2D30", + "j# c #00003C", + "k# c #000015", + "l# c #080808", + "m# c #343443", + "n# c #1E1E30", + "o# c #181831", + "p# c #060606", + "q# c #111111", + "r# c #54545A", + "s# c #121233", + "t# c #303033", + "u# c #00003D", + "v# c #000016", + "w# c #666669", + "x# c #141435", + "y# c #202032", + "z# c #00003E", + "A# c #606032", + "B# c #6C6C6E", + "C# c #303038", + "D# c #2F2F35", + "E# c #090909", + "F# c #8E8E14", + "G# c #B9B915", + "H# c #CCCC1E", + "I# c #DBDB1C", + "J# c #E4E400", + "K# c #DEDE00", + "L# c #BDBD25", + "M# c #828228", + "N# c #676769", + "O# c #171737", + "P# c #00003F", + "Q# c #696953", + "R# c #C4C405", + "S# c #E7E700", + "T# c #E9E900", + "U# c #E8E800", + "V# c #E3E300", + "W# c #DBDB00", + "X# c #D0D000", + "Y# c #AAAA0C", + "Z# c #545409", + "`# c #57575B", + " $ c #292939", + ".$ c #1B1B34", + "+$ c #000017", + "@$ c #97972D", + "#$ c #E5E500", + "$$ c #E2E200", + "%$ c #B8B800", + "&$ c #676751", + "*$ c #3E3E45", + "=$ c #191938", + "-$ c #282835", + ";$ c #000040", + ">$ c #626259", + ",$ c #939300", + "'$ c #C0C000", + ")$ c #DDDD00", + "!$ c #E0E000", + "~$ c #D9D900", + "{$ c #CACA00", + "]$ c #A8A800", + "^$ c #5B5B5D", + "/$ c #0E0E35", + "($ c #2D2D3C", + "_$ c #292936", + ":$ c #5F5F14", + "<$ c #848400", + "[$ c #AAAA00", + "}$ c #E6E600", + "|$ c #D6D600", + "1$ c #8F8F00", + "2$ c #1B1B24", + "3$ c #2F2F3E", + "4$ c #2A2A37", + "5$ c #000041", + "6$ c #4E4E00", + "7$ c #707000", + "8$ c #858500", + "9$ c #A7A700", + "0$ c #BEBE00", + "a$ c #D5D500", + "b$ c #C3C300", + "c$ c #6D6D00", + "d$ c #2B2B38", + "e$ c #000042", + "f$ c #1A1A00", + "g$ c #4D4D00", + "h$ c #6C6C00", + "i$ c #838300", + "j$ c #A4A400", + "k$ c #BCBC00", + "l$ c #CFCF00", + "m$ c #E1E100", + "n$ c #D2D200", + "o$ c #BFBF00", + "p$ c #000011", + "q$ c #1D1D3D", + "r$ c #171700", + "s$ c #474700", + "t$ c #696900", + "u$ c #818100", + "v$ c #A2A200", + "w$ c #BBBB00", + "x$ c #CECE00", + "y$ c #DADA00", + "z$ c #AFAF00", + "A$ c #1E1E3D", + "B$ c #252530", + "C$ c #000043", + "D$ c #121200", + "E$ c #414100", + "F$ c #676700", + "G$ c #808000", + "H$ c #A0A000", + "I$ c #DFDF00", + "J$ c #D8D800", + "K$ c #8D8D00", + "L$ c #1E1E3E", + "M$ c #151525", + "N$ c #0A0A00", + "O$ c #373700", + "P$ c #656500", + "Q$ c #7F7F00", + "R$ c #9F9F00", + "S$ c #C2C200", + "T$ c #525230", + "U$ c #2C2C41", + "V$ c #111124", + "W$ c #000044", + "X$ c #000019", + "Y$ c #343400", + "Z$ c #616100", + "`$ c #7E7E00", + " % c #DCDC00", + ".% c #D1D100", + "+% c #AEAE00", + "@% c #2E2E43", + "#% c #090921", + "$% c #2E2E00", + "%% c #5D5D00", + "&% c #7D7D00", + "*% c #D3D300", + "=% c #7F7F1F", + "-% c #01011F", + ";% c #000045", + ">% c #282800", + ",% c #5A5A00", + "'% c #7C7C00", + ")% c #D4D400", + "!% c #4B4B30", + "~% c #000046", + "{% c #232300", + "]% c #545400", + "^% c #7A7A00", + "/% c #C1C100", + "(% c #A4A412", + "_% c #181800", + ":% c #505000", + "<% c #787800", + "[% c #C3C301", + "}% c #6F6F26", + "|% c #515158", + "1% c #000047", + "2% c #131300", + "3% c #777700", + "4% c #A1A100", + "5% c #C4C401", + "6% c #D7D700", + "7% c #BDBD00", + "8% c #414132", + "9% c #242444", + "0% c #141400", + "a% c #4C4C00", + "b% c #767600", + "c% c #A0A001", + "d% c #C6C602", + "e% c #94941A", + "f% c #5D5D61", + "g% c #000012", + "h% c #000048", + "i% c #747400", + "j% c #C8C803", + "k% c #DBDB01", + "l% c #C4C400", + "m% c #4F4F33", + "n% c #424251", + "o% c #000049", + "p% c #0B0B27", + "q% c #42421A", + "r% c #717100", + "s% c #9D9D02", + "t% c #C9C907", + "u% c #DCDC03", + "v% c #616165", + "w% c #000013", + "x% c #37371B", + "y% c #6A6A00", + "z% c #999904", + "A% c #CDCD0C", + "B% c #DDDD05", + "C% c #E4E401", + "D% c #DCDC01", + "E% c #5A5A2E", + "F% c #6F6F71", + "G% c #00004A", + "H% c #2C2C1B", + "I% c #646400", + "J% c #979707", + "K% c #D3D313", + "L% c #DEDE07", + "M% c #E3E303", + "N% c #D5D501", + "O% c #3A3A4E", + "P% c #23231B", + "Q% c #5E5E00", + "R% c #98980A", + "S% c #DADA1C", + "T% c #DEDE06", + "U% c #E4E403", + "V% c #E1E106", + "W% c #C5C501", + "X% c #5D5D2E", + "Y% c #3B3B50", + "Z% c #00004B", + "`% c #1C1C2F", + " & c #585800", + ".& c #9A9A0E", + "+& c #E3E326", + "@& c #DDDD06", + "#& c #E6E608", + "$& c #DBDB08", + "%& c #3A3A4F", + "&& c #171735", + "*& c #525200", + "=& c #A2A218", + "-& c #EBEB2E", + ";& c #EBEB12", + ">& c #CCCC08", + ",& c #636329", + "'& c #3C3C51", + ")& c #777779", + "!& c #00004C", + "~& c #121236", + "{& c #4B4B00", + "]& c #A5A521", + "^& c #ECEC30", + "/& c #E2E20C", + "(& c #ECEC1A", + "_& c #A9A903", + ":& c #3E3E48", + "<& c #00004D", + "[& c #000014", + "}& c #080837", + "|& c #3E3E2A", + "1& c #91912C", + "2& c #A6A623", + "3& c #B5B532", + "4& c #888825", + "5& c #2C2C3E", + "6& c #070707", + "7& c #00004E", + "8& c #0D0D0D", + "9& c #00004F", + "0& c #32324C", + "a& c #5F5F64", + "b& c #565659", + "c& c #050539", + "d& c #000050", + "e& c #1D1D47", + "f& c #353541", + "g& c #3E3E50", + "h& c #18183E", + "i& c #000051", + "j& c #56565C", + "k& c #3F3F43", + "l& c #000052", + "m& c #1D1D42", + "n& c #272737", + "o& c #01013A", + "p& c #10103D", + "q& c #000053", + "r& c #090944", + "s& c #363643", + "t& c #000054", + "u& c #03031F", + "v& c #151539", + "w& c #171740", + "x& c #242430", + "y& c #000055", + "z& c #040416", + "A& c #0B0B46", + "B& c #2A2A3A", + "C& c #444453", + "D& c #0F0F3F", + "E& c #000056", + "F& c #FEFEFE", + "G& c #22224E", + "H& c #090946", + "I& c #35353A", + "J& c #000057", + "K& c #747479", + "L& c #14143A", + "M& c #1A1A44", + "N& c #47475A", + "O& c #36363F", + "P& c #39393E", + "Q& c #000058", + "R& c #1F1F4E", + "S& c #21214F", + "T& c #29293A", + "U& c #000059", + "V& c #51515F", + "W& c #46464E", + "X& c #131342", + "Y& c #21213B", + "Z& c #1D1D4E", + "`& c #00005A", + " * c #45455A", + ".* c #212135", + "+* c #00005B", + "@* c #76767B", + "#* c #4C4C50", + "$* c #2C2C56", + "%* c #28283A", + "&* c #00005C", + "** c #5B5B62", + "=* c #161645", + "-* c #242453", + ";* c #00005D", + ">* c #000018", + ",* c #48485D", + "'* c #222237", + ")* c #00005E", + "!* c #737379", + "~* c #0A0A4C", + "{* c #2C2C33", + "]* c #303041", + "^* c #00005F", + "/* c #151547", + "(* c #39393F", + "_* c #000060", + ":* c #232339", + "<* c #000061", + "[* c #0E0E47", + "}* c #000062", + "|* c #000063", + "1* c #000064", + "2* c #000065", + "3* c #000066", + "4* c #000067", + "5* c #000068", + "6* c #000069", + "7* c #00006A", + "8* c #00006B", + "9* c #00006C", + "0* c #00006D", + "a* c #00006E", + "b* c #00006F", + "c* c #0D0D2B", + "d* c #2E2E3E", + "e* c #000070", + "f* c #131352", + "g* c #282832", + "h* c #000071", + "i* c #000072", + "j* c #000073", + "k* c #3D3D00", + "l* c #3F3F35", + "m* c #000074", + "n* c #404000", + "o* c #5B5B00", + "p* c #686800", + "q* c #666600", + "r* c #515135", + "s* c #48483F", + "t* c #1D1D5E", + "u* c #46462B", + "v* c #424200", + "w* c #3E3E00", + "x* c #595900", + "y* c #6B6B00", + "z* c #5D5D2B", + "A* c #4C4C3F", + "B* c #000075", + "C* c #404049", + "D* c #636300", + "E* c #626200", + "F* c #363600", + "G* c #0F0F00", + "H* c #272700", + "I* c #303000", + "J* c #5F5F00", + "K* c #6E6E00", + "L* c #6F6F00", + "M* c #373753", + "N* c #000076", + "O* c #515140", + "P* c #8E8E00", + "Q* c #878700", + "R* c #757500", + "S* c #727200", + "T* c #4B4B2B", + "U* c #0D0D00", + "V* c #2C2C00", + "W* c #464600", + "X* c #535300", + "Y* c #737300", + "Z* c #3B3B53", + "`* c #1F1F5F", + " = c #555536", + ".= c #898900", + "+= c #999900", + "@= c #A5A500", + "#= c #9B9B00", + "$= c #8B8B00", + "%= c #888800", + "&= c #20205F", + "*= c #101000", + "== c #323200", + "-= c #494900", + ";= c #797900", + ">= c #7B7B00", + ",= c #7C7C01", + "'= c #333354", + ")= c #000077", + "!= c #101060", + "~= c #434337", + "{= c #828200", + "]= c #929200", + "^= c #ACAC00", + "/= c #B0B000", + "(= c #9D9D00", + "_= c #9A9A00", + ":= c #5E5E40", + "<= c #010140", + "[= c #111100", + "}= c #7E7E02", + "|= c #808004", + "1= c #82820C", + "2= c #888811", + "3= c #7C7C0C", + "4= c #2D2D54", + "5= c #1F1F40", + "6= c #484800", + "7= c #868600", + "8= c #9E9E00", + "9= c #B4B400", + "0= c #B5B500", + "a= c #B6B600", + "b= c #B2B200", + "c= c #959500", + "d= c #282860", + "e= c #06064A", + "f= c #0E0E00", + "g= c #1C1C00", + "h= c #383800", + "i= c #818113", + "j= c #868614", + "k= c #808002", + "l= c #828201", + "m= c #7E7E04", + "n= c #6F6F05", + "o= c #565601", + "p= c #4B4B01", + "q= c #37371E", + "r= c #161637", + "s= c #000078", + "t= c #1E1E1F", + "u= c #3F3F00", + "v= c #606000", + "w= c #989800", + "x= c #A9A900", + "y= c #B3B300", + "z= c #545441", + "A= c #070737", + "B= c #0B0B00", + "C= c #7D7D26", + "D= c #8E8E2D", + "E= c #727203", + "F= c #34341F", + "G= c #02024B", + "H= c #10101F", + "I= c #353500", + "J= c #4F4F00", + "K= c #909000", + "L= c #BABA00", + "M= c #494941", + "N= c #060641", + "O= c #2B2B00", + "P= c #000079", + "Q= c #202000", + "R= c #575700", + "S= c #717107", + "T= c #858519", + "U= c #8C8C20", + "V= c #6D6D04", + "W= c #717104", + "X= c #737304", + "Y= c #787804", + "Z= c #84840B", + "`= c #96960E", + " - c #A0A00F", + ".- c #A0A004", + "+- c #939303", + "@- c #60601F", + "#- c #111156", + "$- c #1D1D00", + "%- c #2D2D00", + "&- c #515106", + "*- c #525207", + "=- c #6B6B1C", + "-- c #5F5F15", + ";- c #434302", + ">- c #404001", + ",- c #444402", + "'- c #4E4E04", + ")- c #686810", + "!- c #6C6C11", + "~- c #555500", + "{- c #39391F", + "]- c #09094C", + "^- c #00007A", + "/- c #09092D", + "(- c #0C0C38", + "_- c #050538", + ":- c #070738", + "<- c #00007B", + "[- c #00007C", + "}- c #00007D", + ". . . . . . . . . . . + + + + + + + + + + + + + + + + + @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ # # # # # # # # # # # # # # # # # # # # # # # # # # # # # $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ $ # # # # # # # # # # # # # # # # # # # # # # # # # # # # # @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ + + + + + + + + + + + + + + + + + . . . . . . . . . . . . . ", + "% % % . . . . . . . . . . . . . . . . + + + + + + + + + + + + + + + + + + + @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ + + + + + + + + + + + + + + + + + + + . . . . . . . . . . . . . . . . % % % % % ", + "% % % % % % % % % % % . . . . . . . . . . . . . . . . . + + + + + + + + + + + + + + + + + + + + + @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ + + + + + + + + + + + + + + + + + + + + + . . . . . . . . . . . . . . . . . % % % % % % % % % % % % % ", + "& & & % % % % % % % % % % % % % % % % % . . . . . . . . . . . . . . . . . . . + + + + + + + + + + + + + + + + + + + + + + + + + @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ + + + + + + + + + + + + + + + + + + + + + + + + + . . . . . . . . . . . . . . . . . . . % % % % % % % % % % % % % % % % % & & & & & ", + "& & & & & & & & & & & % % % % % % % % % % % % % % % % % % . . . . . . . . . . . . . . . . . . . . . . + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + . . . . . . . . . . . . . . . . . . . . . . % % % % % % % % % % % % % % % % % % & & & & & & & & & & & & & ", + "* * * & & & & & & & & & & & & & & & & & % % % % % % % % % % % % % % % % % % % % . . . . . . . . . . . . . . . . . . . . . . . . . . + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + . . . . . . . . . . . . . . . . . . . . . . . . . . % % % % % % % % % % % % % % % % % % % % & & & & & & & & & & & & & & & & & * * * * * ", + "* * * * * * * * * * * * & & & & & & & & & & & & & & & & & & % % % % % % % % % % % % % % % % % % % % % % % . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . % % % % % % % % % % % % % % % % % % % % % % % & & & & & & & & & & & & & & & & & & * * * * * * * * * * * * * * ", + "= = = = * * * * * * * * * * * * * * * * * & & & & & & & & & & & & & & & & & & & & & % % % % % % % % % % % % % % % % % % % % % % % % % % % . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . % % % % % % % % % % % % % % % % % % % % % % % % % % % & & & & & & & & & & & & & & & & & & & & & * * * * * * * * * * * * * * * * * = = = = = = ", + "= = = = = = = = = = = = = * * * * * * * * * * * * * * * * * * * & & & & & & & & & & & & & & & & & & & & & & & % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % . . . . - ; > , ' ) ) ! , ~ { { ] % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % & & & & & & & & & & & & & & & & & & & & & & & * * * * * * * * * * * * * * * * * * * = = = = = = = = = = = = = = = ", + "^ ^ ^ ^ = = = = = = = = = = = = = = = = = = * * * * * * * * * * * * * * * * * * * * * & & & & & & & & & & & & & & & & & & & & & & & & & & & & & & % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % % / ( _ : < [ [ [ < < } | 1 2 3 4 5 6 7 8 % % % % % % % % % % % % % % % % % % % % % % & & & & & & & & & & & & & & & & & & & & & & & & & & & & & & * * * * * * * * * * * * * * * * * * * * * = = = = = = = = = = = = = = = = = = ^ ^ ^ ^ ^ ^ ", + "^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ = = = = = = = = = = = = = = = = = = = * * * * * * * * * * * * * * * * * * * * * * * * * & & & & & & & & & & & & & & & & & & & & & & & & & & & & & & & & & & & & & & & & & & & & & 9 0 a b c d e e e d [ | 2 f g h i j k l m n o p & & & & & & & & & & & & & & & & & & & & & & & & & & & & & & & & & & * * * * * * * * * * * * * * * * * * * * * * * * * = = = = = = = = = = = = = = = = = = = ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ", + "q q q q q ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ = = = = = = = = = = = = = = = = = = = = = * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * & & & & & & & & & & & & & & & & & & & & & & & & & & r s t u < e } } e d v f w x y z A B C D E F G H I J & & & & & & & & & & & & & * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * = = = = = = = = = = = = = = = = = = = = = ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ q q q q q q q ", + "q q q q q q q q q q q q q q q ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ = = = = = = = = = = = = = = = = = = = = = = = = = = * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * K L _ M [ v v v } e 1 x g N O P Q R S T U V W H X Y Z ` * * * * * * * * * * * * * * * * * * * * * * * * * * * = = = = = = = = = = = = = = = = = = = = = = = = = = ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ q q q q q q q q q q q q q q q q q ", + " . . . . . .q q q q q q q q q q q q q q q q q q q ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = * * * * * * * * * * * * * * * * * * * * ..+.@.< e | | | v | w N #.#.$.%.i &.*.k =.-.;.>.X Y ,.'.).* * * * = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ q q q q q q q q q q q q q q q q q q q . . . . . . . .", + " . . . . . . . . . . . . . . . .q q q q q q q q q q q q q q q q q q q q q ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = !.~.u [ v 1 2 1 | f N {.].^./.T 4 (.R C l _.;.>.:.Y ,.<.[.}.= = = = = = = = = = = = = = = = = = = = = = ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ q q q q q q q q q q q q q q q q q q q q q . . . . . . . . . . . . . . . . . .", + "|.|.|.|.|.|.|. . . . . . . . . . . . . . . . . . . . .q q q q q q q q q q q q q q q q q q q q q q q q ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ = = = = = = = = = = = 1.2._ M } 1 f f 2 1 x {.y y 3.<.4 4.j &.5.6.7.;.8.:.Y 9.<.[.0.^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ q q q q q q q q q q q q q q q q q q q q q q q q . . . . . . . . . . . . . . . . . . . .|.|.|.|.|.|.|.|.|.", + "|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|. . . . . . . . . . . . . . . . . . . . . . .q q q q q q q q q q q q q q q q q q q q q q q q q q q q q q ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ [email protected] v 2 w w f w #.y c.d.e.9.f.g.4 B S D 7.h.i.:.Y ,.j.[.k.^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ ^ q q q q q q q q q q q q q q q q q q q q q q q q q q q q q q . . . . . . . . . . . . . . . . . . . . . .|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.", + "l.l.l.l.l.l.l.l.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|. . . . . . . . . . . . . . . . . . . . . . . . . .q q q q q q q q q q q q q q q q q q q q q q q q q q q q q q q q q q q q q q q q q q q q q q q q ; +.u e 1 w x x w g ^.d.e.m.n.o.A p.4.(.*.D 7.h.i.q.Y r.j.s.t.q q q q q q q q q q q q q q q q q q q q q q q q q q q q q q q q . . . . . . . . . . . . . . . . . . . . . . . . .|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.l.l.l.l.l.l.l.l.l.l.", + "l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .q q q q q q q q q q q q q q q q q q q q q q q q q q q q u.t M v f x g g g {.3.m.O v.$.8.A w.g.j *.T m h.i.q.Y 9.3 s.x.q q q q q q q q q q q q . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.y.", + "y.y.y.y.y.y.y.y.y.y.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .z.A.d 1 w N N N #.y e.n.B.B.s.V C.D.p.j R T 7.h.i.:.Y ,.E.F.G. . . . . . . . . . . . . . . . . . . . . . . . . . . . .|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.y.y.y.y.y.y.y.y.y.y.y.y.", + "H.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|. . . . . . . . . . . . . . . . . . . . .I.J.K.e f g #.#.#.^.d.v.L.M.N.3 C f.C.p.4 R T 7.h.i.:.Y O.P.F.Q. . . . . .|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.H.H.H.", + "H.H.H.H.H.H.H.H.H.H.H.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.R.S.M v x N {.{.{.3.O L.T.T.U.V.4.i C.w.4 &.T 7.h.8.X W.O.3 F.X.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.|.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.H.H.H.H.H.H.H.H.H.H.H.H.H.", + "Y.Y.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.|.|.|.|.|.|.|.|.|.Z.t d 2 g {.].].^.e.B.T.$.`. +.+f.i f.D.4.&.k 7.;.8.++@+#+P.$+%+l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.Y.Y.Y.Y.", + "Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.&+A.e w #.].y ^.c.v.N.`. +$+F.*+A Q i C.g.B k 7.;.8.=+@+-+;+`.>+l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.l.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.", + ",+,+,+,+Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.'+u v g {.y 3.y O L.$.)+F.s.!+~+A A Q f.w.(.k 7.W >.*+@+{+/.$.]+y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.,+,+,+,+,+,+", + ",+,+,+,+,+,+,+,+,+,+,+,+,+,+,+Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.y.( c 2 #.^.3.c.c.B.T.$+s.[.^+3 7.A /+A Q C.j k _.W H (+_+:+<+T.y.y.y.y.y.y.y.y.y.y.y.y.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+", + "[+[+[+[+[+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.}+t [ w {.3.c.d.m.N.`.s.!+<+/.{+S /+|+|+/+i 4.k _.W P o.1+2+!+3+H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+[+[+[+[+[+[+[+", + "[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.H.4+@.v g ^.c.e.e.n.$.F.!+;+;+P.r.4.|+5+6+6+/+p.C -.G 7+8+.+<.[.k.H.H.H.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+", + "9+9+9+9+9+9+9+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.0+u 2 #.3.d.m.m.N.$+^+;+E.3 j.a+A |+b+c+d+6+D.C -.e+f+z 9.3 s.g+Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.Y.,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+9+9+9+9+9+9+9+9+9+", + "9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+h+c w ].c.m.O n.`.s./.3 <.'.'.7+/+5+d+i+j+k+i 5.V l+m+n+h j.F.o+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+", + "p+p+p+p+p+p+p+p+p+p+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+q+e g y e.O v.N.F.<+3 2+r+r+:+e+|+b+i+s+t+u+A S V v+X Y V.P.)+w+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+,+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+p+p+p+p+p+p+p+p+p+p+p+p+", + "p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+x+_ | #.c.m.v.n.$.^+P.'.y+y+{+#+E z+c+A+B+B+t+z+*.F i.=+W.-+<+ +C+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+D+", + "D+D+D+D+D+D+D+D+D+D+D+D+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+[+E+u f ^.e.v.B.M.)+;+<.y+F+F+F+r.R 5+G+H+B+B+B+6+*.n >.(+_+y+^+$.I+[+[+[+[+[+[+[+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+D+D+D+D+D+D+D+D+D+D+D+D+D+D+", + "J+J+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+K+< g y m.n.L.U.[.3 :+#+O.O.O.W.Q 6+u+B+B+B+B+L+S %.P 8+M+2+[.T.N+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+9+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+J+J+J+J+", + "J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+O+e #.d.v.B.M.$+;+2+F+h 9.9.9.=+A L+P+B+B+B+B+c+C h.7+z Q+j.F.M.R+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+", + "S+S+S+S+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+T+1 ].e.n.M.T.s.j.{+h .+M+.+Q+8./+c+H+B+B+B+B+d+k ;.m+Y h P.$+n.U+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+p+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+S+S+S+S+S+S+", + "S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+b w y O L.N. +<+r+V.Q+_+V+V+1+n /+k+B+B+B+B+B+c+D G X @+#+;+ +m.D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+", + "W+W+W+W+W+W+W+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+D+X+c N c.n.M.U.s.E.-+r._+n+Y W.@+=.|+Y+B+B+B+B+B+b+=.v+=+V+-+^+$.0.D+D+D+D+D+D+D+D+D+D+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+W+W+W+W+W+W+W+W+W+", + "W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+Z+e {.e.B.N.`.!+'.h V+z o.a+z z f.|+j+B+B+B+B+B+|+m >.o..+:+F.T.`+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+J+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+", + " @ @ @ @ @ @ @ @ @ @W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+.@1 y O M.T.)+P.{+Q+Y *+++*+o.++f.|+u+B+B+B+B+B+i V +@z r.<.$+M.@@S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+ @ @ @ @ @ @ @ @ @ @ @ @", + " @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+#@w c.n.N.U.^+<.O._+(+q.m+X *++@f.|+$@B+B+B+B+B+g.%.m+@+h E.`.n.%@S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+S+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @", + "&@&@&@&@&@&@&@&@&@&@&@&@&@&@ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+*@N e.L.T.`./.:+Q+z :.P P f+X e+C./+$@B+B+B+B+s+R G ++1+F+<+T.O =@W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@", + "-@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+W+;@d ].O M.U.F.3 F+_+=+P v+i.P m+k D.A j+B+B+B+B+d+T ~+o.Q+:+s.M.d.>@W+W+W+W+W+W+W+W+W+ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@-@-@-@", + "-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @,@v y v.T.$.!+2+,.z f+l+;.e+8.7+B p.i G+B+B+B+B+A 7.P z ,.2+$+B.3.'@ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@", + ")@)@)@)@)@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@!@f d.B.U. +P.{+.+=+8.~@F h.v+H p.4 D.c+B+B+B+u+4 ~@q.@+V.3 U.v.].{@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@)@)@)@)@)@)@)@", + ")@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@]@N m.M.$.s.<.V.@+f+W _.m F G ~+(.(.g.5+A+B+^@z+C v+*+M+{+/.M.m.N /@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@&@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@", + "(@(@(@(@(@(@(@(@(@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@_@].v.T.`.!+:+9.8+8.V 6.6.7.h.;.B *.(.i d+j+c+p.m H a+9.r+[.n.c.w :@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@-@<@[@[@}@|@1@2@3@4@5@6@-@-@-@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@(@(@(@(@(@(@(@(@(@(@(@", + "(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@-@-@-@-@-@-@-@-@-@-@-@-@-@-@7@e 3.n.U. +E.F+M+X W =.S 5.=.n _.*.C *.g./+z+C.5.h.q.W.O.j.$+O y | -@-@-@-@-@-@-@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@8@9@0@a@b@c@d@e@f@g@h@i@j@k@k@l@m@n@o@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@", + "p@p@p@p@p@p@p@p@p@p@p@p@p@p@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@q@2 d.L.$.s.'.O.W.P F 5.j &.D V 6.k 6.k &.g.p.R m ~+(+M+-+;+U.d.{.[ )@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@)@r@a@s@t@u@d@e@f@h@h@j@k@v@w@x@x@y@z@y@y@A@B@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@", + "p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@C@g m.N. +<+y+r.8+v+U j D.4.C _.l =.E =.k *.S =.h.7+n+9.r+!+L.3.g D@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@E@F@G@d@d@e@H@I@J@I@h@k@v@w@x@K@L@M@K@N@O@O@O@O@O@P@(@(@(@(@(@(@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@Q@", + "Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@R@#.n.T.)+E.F+1+++h.C C.w.D.*.m l 7.-.7.U l E n i.++V+V.j.F.O #.w S@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@(@T@U@g@g@j@j@j@V@l@L@k@k@x@N@y@M@N@W@X@N@Y@Z@`@`@W@`@ #.#+#p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@", + "@#@#@#@#Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@##^.B.$.s.'.h W.7+_.*.h.~+8.X f+_+Q+r.r..+.+r.Y r.y+-+{+;+`.r+8.8._+$#p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@p@%#&#k@w@x@z@`@K@K@L@*#W@L@O@=#W@X@W@*#-#W@*#;#Y@>#;#-#-#W@Z@,#Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@@#@#@#@#@#@#", + "@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@p@p@p@p@} d.M. +<+:+y+Q+y+h (+v+;.F ;.++a '#'#'#'#'#'#'#'#'#'#'#>#d.v+_.6.~@(+)#Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@!#z@y@X@O@N@ #;#~#-#-#{#]#;#>#^#]#>#/#]#~#(#_#{#;#*#:#;#/#;#>#-#<#Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#", + "[#[#[#[#[#[#[#[#[#[#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@d ]#}#W@'#'#'#'#'#v i.l+_.S *.n @+|#'#'#'#1#1#2#1#1#'#'#'#r.h.(./+4 G .+3#Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@4#L@ #`@>#5#(#;#:#2.6#=#^#t 7#{#s 8#9#|#S.0 ~#|#]#_#:#0#=#~#:#;#/#Y@a#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#[#[#[#[#[#[#[#[#[#[#[#[#", + "[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#q '#'#'#'#'#'#'#'#E.e+G l D.z+j >.M.'#1#2#b#c#d#d#c#b#1#*#i.V A s+d+k *+e#@#@#@#@#@#@#@#@#@#@#@#@#@#f#`@-#>#;#5#g#^#h#6#S._ 0 2.: M ( 8#} A.+.c @.!.( a 6#6#!.|#h#6#=#~#*#i#@#@#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#", + "j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#@#D+k#'#'#'#'#'#'#'#'#1+h.G k z+u+L+E 9.`@b#d#l#s@b@b@s@l#c#9#F V 6+B+^@p.>.{+m#@#@#@#@#@#@#@#@#@#@#@#n#>#(#_#:#{#}#7#_ J.0 8#| M _ v {.} | c.x } ].f K.e < s 2.+.9#0#!.h#h#_#5#o#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#", + "j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[#[# .'#'#'#'#'#'#'#'#2#=+-.l+k b+H+P+(.z b p#s@c@J@h@j@h@q#t@}#D %.|+c+_.*.W h r#[#[#[#[#[#[#[#[#[#[#s#*#~#:#}#J.0 !.L M e A.K.N y x ^.N.v.m.`.M.m.U.O {.e.g : [ : 7#7#s ^#^#}#_#t#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#", + "u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#v#'#'#'#'#'#'#'#2#p#9.T i.q.v.9.6+D.*+g s@q#j@M@Z@Y@`@z@g@[email protected]+6.< e@[email protected]#j#j#j#j#j#j#j#j#j#x#{ :#|#^#g#+.u t S.< ^.N 2 d.`.T.T.<+/.[.'.<.;+2+<+)+[.n.y d.| t A.S.!.7#]#|#_#y#j#j#j#j#j#j#j#j#j#j#j#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#", + "z#z#z#z#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#j#S+'#'#'#'#'#'#'#'#c#u@x (.8+J@L@Y@: C.++{.u@i@N@A#2.S.J.}#`@h@r.+@M+d@V@f@)+h B#j#j#j#j#j#j#j#j#j#C#}#^#2.2.J.a e w d e c.U.N.U.;+<.P.#+Q+,.z (+a+*+W.V+9.;+<+$+].g x A.L b.!.^#|#D#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#z#z#z#z#z#z#", + "z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#u#|.'#'#'#'#'#'#'#1#d#c@*#i.,.d#f@E#1#_.z f d@F#G#H#I#J#K#L#M#v@c@q.=+'#'#'#,.{+N#u#u#u#u#u#u#u#u#O#]#g#0 S.K._ A.[ d.e.y n.!+3 E.y+9.1+(++@e+V E l l m F v+o.V+2+`.U.3.: A._ s 0 0#:#u#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#", + "P#P#P#P#P#P#P#P#P#P#P#P#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#u#u#u#u#u#u#@ '#'#'#'#'#'#'#'#2#l#J@$+8+_ '#'#L -.O.Q#R#J#S#T#U#S#V#W#X#Y#Z#$+Y V+3 ^+a+E.`#z#z#z#z#z#z#z#z# $!.2.+.~.d w } v d.$+^+s.E.#+r.1+++i.V T B g.C.Q A Q C.4 6.W z ,.E.v.].#.c b.J.g#|#.$z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#P#P#P#P#P#P#P#P#P#P#P#P#P#P#", + "P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#z#+$'#'#'#'#'#'#'#'#'#1#c# #/.,.v.)+f+o.$+@$X#K#J#S#U#U##$$$W#X#%$&$2+V+X o.#+T.*$z#z#z#z#z#z#z#=$2.L t 8#b d y O 3.O F.:+#+V.V+(+H n 6.R 4 w.i |+5+6+6+5+|+i (.-.f+O.!+ +d.M ~.a 0 ^#-$P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#", + ";$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#+ '#'#'#'#'#'#'#'#'#'#'#'#0#$.r+9.r./.>$,$'$X#)$J#S#U#S#J#!$~${$]$M ^+2+<. +^$/$P#P#P#P#P#P#P#($b.S.K.} [ } {.$.[.^+P.#+_+8+X l+7.T *.j w.i |+6+L+c+c+L+6+|+f.j 7.++9.;+e.N f b L g#_$P#P#P#P#P#P#P#P#P#P#P#P#;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$", + ";$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#P#= '#'#'#'#'#'#'#'#'#'#'#'#'#`@< d.^.t :$<$[$'$X#)$J#S#S#}$V#K#|${$1$}#v f ~.2$P#P#P#P#P#P#P#P#3$S.A.u 2 c.^.].U.E.,.V+@+*++@G 7.6.5.B 4.C.A 5+L+d+k+k+d+b+z+Q p.&.n o.'.s.N.].A.( 7#4$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$", + "5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$;$p+'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#H@c@'#6$7$8$9$0$X#)$J#S#S##$$$)$a$b$c$1#c#'#p+;$;$;$;$;$;$;$;$]@A.K.[ 1 e. +`.)+j.1+q.H i.;.7.=.T *.(.p.i |+6+c+k+G+k+d+b+z+i p.B D >.M+'.`.g c _ 2.d$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$", + "e$e$e$e$e$e$e$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#f$g$h$i$j$k$l$)$J#}$}$J#m$W#n$o$6$'#p$5$5$5$5$5$5$5$5$q$_ K.v | w 3. +2+,.V+=+~+;[email protected] 6.C R j w.i |+6+L+d+k+d+c+6+/+f.g.&.k %.o.y+$+B.y [ L Z+5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$e$e$e$e$e$e$e$e$e$", + "e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$5$p$'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#r$s$t$u$v$w$x$)$J#}$}$J#!$y$X#z$'#. 5$5$5$5$5$5$5$5$A$K.M e {.3.e.T.j.n+W E U _.7.E =.D C R j p.f./+5+b+L+L+L+6+|+Q w.4 *.D _.m+,.j.)+3.M ( B$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$", + "C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$q '#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#D$E$F$G$H$w$l$K##$S#}$V#I$J${$K$D+e$e$e$e$e$e$e$e$L$: d } f e.U.[.r+W.%.R 5.=.U =.6.T C R (.g.D.Q /+z+5+5+z+|+Q C.g.B C l _.8.V+'.$.{.e a M$e$e$e$e$e$e$e$e$e$e$e$e$e$e$e$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$", + "C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$e$e$'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#N$O$P$Q$R$w$X#I$}$S#}$V#K#a$S$T$C$C$C$C$C$C$C$C$U$< | 1 f #.B.3 o.E p.A B 6.l l 6.T C *.B 4 p.C.i A A A Q f.D.g.(.S D E V i._+r+!+T.y J.V$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$", + "W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$C$X$'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#Y$Z$`$R$k$n$!$}$S#}$$$ %.%+%C$C$C$C$C$C$C$C$@%e 1 #.#.g y T.1+&.Y+L+4 D 6.l 6.D k 5.R B 4 g.w.D.C.C.D.p.4 B S T =._.%.i.n+#+E.U.{.2.#%W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$", + "W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$,+'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#$%%%&%R$0$*%m$S#S#}$m$y${$=%W$W$W$W$W$W$W$W$v f g y y #.v.F+%.$@i+j T 6.l 6.6.T C S R B j 4 4.4.4.4 (.&.S T =.7.F h.>.W.-+^+O 1 (#-%W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$;%", + ";%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$W$p$'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#>%,%'%H$o$)%$$S#S##$!$|$'$!%W$W$W$W$W$W$W$G.w N {.y ^.^+(+&.t+i+(.T 6.l l l 6.D k 5.S R &.&.&.&.R S C D =.7.F ~@G P @+#+3 $+g t@S+;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%", + "~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;% .'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#{%]%^%H$/%|$V#S#S#J#K#.%(%;%;%;%;%;%;%;%@@N y ^.].].n.V.*.u+j+4 T 6.=.=.=.=.l 6.D k C C 5.5.C k D l U 7.F ~@W v+m+_+V.j. +M d#;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%;%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%", + "~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%Q@'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#_%:%<%H$[%J$V#S#S#V#W#{$}%~%~%~%~%~%~%|%#.y e.c.].3.j.h.Y+A+j D l =.U E E U U =.l 6.6.6.6.l =.U m -.F ~@W l+8.=+.+:+s.]./#'#~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%", + "1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%~%$ '#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#2%6$3%4%5%~$J#S#}$m$6%7%8%~%~%~%~%~%9%{.c.d.e.y $..+T L+6+g.D =.E m m 7.7.m m E E E E E m 7._.V n h.W l+~++@8+M+#+E.u '#* 1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%", + "1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%[+'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#0%a%b%c%d%y$J#S##$I$X#e%1%1%1%1%1%1%f%m.n.m.y 3.F.=+g.i+w.D U m 7._._.-.-._._._._._.-.V F n ~@;.W l+~+>.X W.r.y+L.'#g%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%1%", + "h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#D$s$i%c%j%k%J#}$J#y$l%m%h%h%h%h%h%n%m.M. +T.O +W.C /+w.T E 7._.V V F F F F F F n n %[email protected] G l+~+8.7+(+_+#+$+-#'#* h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%", + "o%o%o%o%o%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%* '#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#p%q%r%s%t%u%J#}$m$n$j$h%h%h%h%h%h%v%M.U.$+)+E.=+F p.g.C m _.V F n n %.%.%.%.~@~@h.;.W G e+v+~+8.P ++z V+r+u '#'#9+h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%h%o%o%o%o%o%o%o%", + "o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%p+'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#w%o%o%x%y%z%A%B%V#C%D%l%E%o%o%o%o%o%o%F% + +`.j.,.P E S k E V F n %.~@[email protected].;.W W G e+l+v+~+8.H q.o.V+r+d '#'#y.o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%", + "G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%o%'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#* o%o%o%H%I%J%K%L%V#M%N%j$o%o%o%o%o%o%O%U.E.:+2+@+i.V U =.m V n ~@~@h.;.;.W W G G e+l+l+v+i.8.H m+=+8+#+< '#'#= G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%", + "G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%w%'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#-@G%G%G%G%P%Q%R%S%T%U%V%W%X%G%G%G%G%G%G%Y%$+3 9.o.q.+@~+%.V F %.h.;.;.W W G G e+l+l+v+~+i.8.P f+++z h t '#'#'#-@G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%G%", + "Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%= '#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#-@Z%Z%Z%Z%Z%`% &.&+&@&#&$&j$Z%Z%Z%Z%Z%Z%Z%%&$+j.F+W.=+>.G ;.h.h.;.W W G G e+l+l+v+v+~+i.>.P m+:.*+F.'#'#'#w%D+Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%", + "Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%D+'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#w%Z%Z%Z%Z%Z%Z%Z%&&*&=&-&T%;&>&,&Z%Z%Z%Z%Z%Z%Z%'&)&r+M+++7++@>.l+G G G e+e+l+l+v+v+~+i.8.H +@7+++1+N '#'#'#w%u#Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%Z%", + "!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#w%!&!&!&!&!&!&!&!&~&{&]&^&/&(&_&!&!&!&!&!&!&!&!&!&:&;+9.z =+P i.~+~+l+l+l+v+v+~+~+i.>.>.+@m+Y T.'#'#'#'#w%H.!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&!&", + "<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&[&'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#@ <&<&<&<&<&<&<&<&<&}&|&1&2&3&4&5&<&<&<&<&<&<&<&<&<&(@d F.n+m+m+f+8.~+i.i.~+i.8.8.H +@+@,.K.'#'#'#'#[&Y.<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&", + "<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&Y.'#'#'#'#'#'#'#b#b#2#1#2#c#6&t@u@b@l#b#'#'#'#'#[&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&J+*#y y+q.+@7+m+H >.f+8+(+<.: '#'#'#'#'#'#[&J+<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&<&", + "7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&'#'#'#'#'#c#8&V@O@`@y@g@q#J@O@{#2.g#W@g@p#1#'#'#'#S+7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&P#)&,.#+0#}#K.]#'#'#'#'#'#'#'#'#'#'#q p@7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&", + "7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&;$'#'#'#'#1#p#J@Z@g#L 6#N@j@V@-#0 b t {#l@E#2#'#'#'#,+9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&0&O._.h.O K@'#'#'#'#'#'#'#'#[&q q p@7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&7&", + "9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&S+'#'#'#'#2#l#i@(#s +.^# #V@V@Z@g#A.t h#v@s@2#'#'#'#'#9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&a&X k W.b&c&;$9&S+q S+p@S+p@9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&9&", + "9&9&9&9&9&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&+ '#'#'#'#2#s@l@=#b.a ]#W@V@V@ #^#t a h#z@b@b#'#'#'#'#5$d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&e&E.n 6.U.f&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&9&9&9&9&9&9&9&", + "d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&Q@'#'#'#'#'#b#t@x@h#S.t g#W@V@j@X@}#( ( |#M@u@b#'#'#'#'# .d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&g&q.l 8.u h&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&d&", + "i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&e$. '#'#'#'#'#c#d@K@}#a ~.0#Z@V@0 e /.8+*+8+1+E.+.1#'#'#'#'#@#i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&j&8+6.O.k&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&", + "i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&[+'#'#'#'#'#1#p#d@y@5#]#0#;#S.2+l 4.f.Q Q f.w.j C v+<+'#'#'#k#l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&m&f *+8.e.n&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&i&", + "l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&o&s@'#'#'#'#'#2#l#e@x@>#=#a P.=.4.Q z+5+5+5+|+A f.g.&.6.;.`.'#'# @l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&B$v V.y+A.p&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&", + "l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&C$v@X@'#'#'#'#'#c#t@i@N@=#| X 5.w.A z+5+6+b+L+L+b+|+f.4.*.6.n .+S.'#C$q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&r&>#e r+{.s&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&l&", + "q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&u&X@'#'#'#'#'#1#p#c@l@W@| i.C g.i |+z+6+c+Y+$@P+P+u+c+Q 4 C E h.q.8#l.t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&v&6#2 M.+.w&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&q&", + "t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&-@x@z@'#'#'#'#'#b#l#q#z@M 7+l j f.|+5+6+L+Y+P+B+B+B+B+H+Y+i B l V l+=+t j#t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&x&s O | ~ t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&q&", + "t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&;%z&X@2#'#'#'#'#1#b#t@[email protected]+-.*.C.5+L+b+b+k+A+B+B+B+B+B+B+B+k+w.C 7.;.H z b.y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&A&*#K.L.~.B&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&t&", + "y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&p+L@V@'#'#'#'#'#2#p#c@~#F+e+6.p.b+i+Y+d+d+Y+t+B+B+B+B+B+B+B+^@|+&.E ~@i.:.Q+C&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&4#]#1 1 !.D&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&", + "y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&~%c#/#u@'#'#'#'#'#b#E#[email protected] &.b+P+H+A+G+k+j+F&B+B+B+B+B+B+B+B+c+4.l %.v+7+z r+G&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&H&O@a m.b I&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&y&", + "E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&y.X@k@'#'#'#'#'#1#c#b@s r.i.U A H+B+B+^@Y+k+j+F&B+B+B+B+B+B+B+B+Y+p.T n v+7+(+.+K&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&L&}#[ {.J.M&J&J&J&J&J&J&J&J&J&J&J&J&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&E&", + "E&E&E&E&E&E&E&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&z#c#/#s@'#'#'#'#'#2#p#[email protected] R $@B+B+B+^@G+c+G+^@B+B+B+B+B+B+B+B+Y+D.k F v++@*+_+y+N&J&J&J&J&J&J&J&J&J&J&J&J&J&J&O&( ].d P&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&E&E&E&E&E&E&E&E&E&", + "J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&& w@W@'#'#'#'#'#'#b#l#2.F+q.n Q B+B+B+B+s+c+6+L+u+B+B+B+B+B+B+B+F&d+p.T F v+7+=+W.O.)+R&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&S&d 3.c.+.T&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&J&", + "Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&P#c#(#s@'#'#'#'#'#1#c#k@]..++@U G+B+B+B+B+j+5+A z+d+A+B+B+B+B+B+B+A+6+4.6.n ~+f+*+Y 9.3 V&U&U&U&U&U&U&U&U&U&U&U&U&W&[.:+e.9#X&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&", + "Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&* K@Z@'#'#'#'#'#'#2#d#{#[.@+>.5.^@B+B+B+H+c+i D.f.|+d+A+F&B+B+B+s+k+i R U h.8.m+(+W.r.2+M.Y&U&U&U&U&U&U&U&U&U&U&Z&n.8.F+w I&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&Q&", + "U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&p@b#Y@e@'#'#'#'#'#'#b#6&u <.n+~+j F&B+B+B+$@/+p.4 4 D./+b+G+u+u+i+c+/+4 k _.G H :.o.@+r.y+s.b.* ;$`&`&`&`&`&`&`&`& *(+7.Q+d .*`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&U&", + "U&U&U&U&U&U&U&U&U&U&U&U&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&+*= h@X@c#'#'#'#'#'#1#c#x@^.y+a+e+p.B+B+B+B+L+w.B *.*.(.g.f./+z+5+|+f.4.S U %.v++@++a+_+9.{+<+^.'#'#;$`&`&`&`&`&`&`&@*=.~@;+#*`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&U&U&U&U&U&U&U&U&U&U&U&U&U&U&", + "`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*5$2#z@d@'#'#'#'#'#'#2#d#=#v.{+a+G p.s+B+H+Y+A B C D D C *.(.4.p.p.4.B 5.=.V G >.m+*+n+V+,.{+;+v.|#'#'#S++*+*+*+*+*$*.+5.X {.%*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&`&", + "`&`&`&`&+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*G%= c@L@2#'#'#'#'#'#'#b#p#s T.{+a+G 4 k+s+Y+|+4 k =.m m U 6.T C S S 5.D U V h.~+7+X 8+W.M+h y+/.M.A.'#'#'#Y.&*&*&*&***>.m .+8#=*&*&*&*&*&*&*&*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*`&`&`&`&`&`&", + "+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*@#'#f@i@'#'#'#'#'#'#1#c#E#K.`.F+Y v+*.Q 6+A 4.C U -.F F V _.m U U U E _.n ;.v+P :.(+n+V+Q+V.r+;+T.x i@'#'#'#e$&*&*-*^+m 8.U.P&&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*+*", + "&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*Z%>*1#e@E#'#'#'#'#'#'#2#d#j@| +{+@+>.l j w.(.5.U F ~@;.W ;.~@%.n F n %.h.e+~+P q.=+a+W.M+,.F+2+/.U.#.~#'#'#'#'#e$;*,*Y =._+#.'*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*", + "&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*)*)*)*)*)*)*)*)*)*)*)*)*[#'#1#c@'#'#'#'#'#'#1#b#p#z@w +y+V++@V T C D E n W e+v+~+v+l+e+G G e+l+~+8.+@q.=+8+Y V+Q+O.{+<.<+U.{.g#'#'#'#'#'#C$!*7.G /.C+~*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*", + ";*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*!&'#'#6&s@'#'#'#'#'#'#1#b#6&;#w U.2+Q+X l+F 7.-.~@G ~+>.H H H >.>.8.8.>.H +@f+:.=+8+Y _+.+,.#+:+3 ^+T.].@.'#'#'#'#'#{*.+R o.].]*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*;*", + ";*;*;*;*;*;*;*;*;*;*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*[#'#'#s@'#'#'#'#'#'#'#2#c#l#{#w T.<.,.8++@l+W G v+8.P 7+f+f+f+f+f+f+f+f+q.:.++(+a+Y _+M+9.V.{+'.P.s.N.].[ '#'#'#'#'#: >.7.O.~./*^*^*^*^*^*^*^*^*^*^*^*^*^*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*;*;*;*;*;*;*;*;*;*;*;*;*", + ")*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^ '#1#1#'#'#'#'#'#'#1#2#d#E#^#f M.E.#+@++++@H H +@f+q.:.X X X X X X ++++=+(+8+z Y _+M+9.O.F+:+j./.)+L.{.d l@'#'#'#[email protected].+@ +(*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*)*", + ")*)*)*)*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*W$'#'#1#'#'#'#'#'#'#'#1#2#p#s@]#1 O <+:+Q+n+*+:.:.X ++*+(+(+(+(+(+(+o.o.8+a+z Y @+V+.+9.O.F+y+'.E.!+ +n.#.< /#'#'#'#s W.U Q+g :*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*)*)*)*)*)*)*", + "^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*q '#'#1#'#'#'#'#'#'#'#1#b#p#t@]#v d.s.<.O.1+Y 8+o.8+8+a+a+z z z z z n+n+Y W.@+V+M+Q+9.O.F+y+2+3 /.s.U.v.g c _#'#'#'#c.m ~+P.a [*<*<*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*^*", + "_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*7&'#'#1#1#'#'#'#'#'#'#'#1#b#p#t@^#e y +P.y+,.M+_+W.W.W.W.W.@+@+@+@+@+@+_+V+1+M+Q+r.,.O.F+y+r+j.P.!+$+N.e.w M }#'#'#[email protected] n+3.*#<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*^*", + "_*_*_*_*_*_*_*_*_*_*_*_*_*_*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*u#'#'#2#'#'#'#'#'#'#'#'#1#b#p#t@6#< g N.[.<.-+h Q+M+M+1+1+1+1+1+1+1+1+M+.+.+Q+9.,.O.#+-+y+2+<.E.<+s.`.B.3.2 u ]#'#'#u H _.{[email protected]@9&}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*", + "<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*|*z#'#2#b#'#'#'#'#'#'#'#'#1#2#d#s@=#M 2 v.$+;+2+-+O.,.9.r.r.r.r.r.r.r.r.9.9.,.h O.V.F+{+:+2+<.E./.[.$+T.O ^.v K.]#'#M@ +6.+@$+:#'#u#}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*<*", + "<*<*<*<*<*<*<*<*<*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*q '#8&I@'#'#'#'#'#'#'#'#'#2#d#E#5#b e c.T.[.E.2+y+F+#+V.O.h h h h h O.O.O.V.#+F+{+y+r+'.j.P./.^+)+$.L.d.N e @.]#'#2..+D O.w L@'#)@|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*<*<*<*<*<*<*<*<*<*<*<*", + "}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*'#2#H@6&'#'#'#'#'#'#'#'#'#2#c#6&K@_ c {.v. +!+E.'.:+y+{+F+F+F+#+#+#+F+F+-+-+{+:+r+2+<.3 P.<+^+F.`.N.v.y w c 8#6#'#y ~+H !++.u@'# .1*1*1*1*1*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*}*", + "}*}*}*}*}*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*i&'#d@l@'#'#'#'#'#'#'#'#'#'#1#b#p#s@a u f d.N.)+!+P.j.'.r+:+:+y+y+y+y+y+:+:+r+2+'.<.j.E.;+<+[.)+ +T.n.e.{.1 K.~.|#[email protected] @+c.~#'#'# .1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*}*}*}*}*}*}*}*", + "|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*P#b#k@t@'#'#'#'#'#'#'#'#'#'#1#2#d#6&7#@.[ ].O U.)+^+;+E.3 <.'.'.2+2+2+'.'.'.<.j.3 E.;+<+!+s.)+`.T.B.m.y x e A.+.~#K.P -.'.< h@'#E# .2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*|*", + "|*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*3*3*l&'#d@k@'#'#'#'#'#'#'#'#'#'#'#'#1#b#d#:#t u 2 3.B.U.)+[.!+/.P.E.3 3 3 3 3 3 E.E.P.;+<+!+[.F.$+`.T.L.O c.#.2 c ~.J.}#$.l q.U..@E#I@f@p+2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*|*|*|*", + "1*1*1*1*1*1*1*1*1*1*1*1*1*1*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*D+b#v@6&'#'#'#'#'#'#'#'#'#'#'#'#'#2#b#y@+._ e N e.L.U.$+s.[.!+<+/.;+;+;+;+/./.<+!+^+[.s.)+ +$.N.L.v.d.].w e K.+.2.S.,.C h | y@J@c#'#(@3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*1*", + "2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*4*4*4*4*4*4*4*4*4*4*4*4*4*4*q&'#J@I@'#'#'#'#'#'#'#'#'#'#'#'#'#'#1#1#2#0 a u 1 ].e.n.T.`.$+F.s.[.^+^+^+^+^+^+[.s.F.)+ +$.U.M.B.O d.^.g 1 c 8#b.!.d.8.f+`.;#f@'#'#'#;$4*4*4*4*4*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*2*", + "2*2*2*2*2*2*2*2*2*2*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*|.8&j@c#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#1#=#b.8#c w ].e.n.M.U.`. +$+)+)+F.F.)+)+$+$+`.$.U.N.L.n.O c.^.N 2 [ b +.2.J.h S [.j@d#'#'#'#'#o%4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*2*2*2*2*2*2*2*2*2*2*2*2*", + "3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5$2#J@6&'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#x@2.( @.} x ^.d.v.B.M.T.U.$.$.`.`.$.$.$.U.T.N.L.n.O e.c.].N f } : t s g#d 7+,.w@'#'#'#'#'#'#G%5*5*5*5*5*5*5*5*5*5*5*5*5*5*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*3*", + "3*3*3*3*3*3*3*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*y&# s@d@'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#0#s +.K.v x {.c.e.v.B.L.M.M.N.N.N.N.M.L.B.n.O m.d.y ].g f } c A.( 0 0 U.;+y@'#'#'#'#'#'## 5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*3*3*3*3*3*3*3*3*3*", + "4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*e$2#q#b## '#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#Z@g#s ~.M } w #.^.c.e.O v.v.n.n.n.n.v.O m.e.d.3.^.#.g 2 } < b a 7#]#!.a q#'#'#'#'#'#'#'#J+6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*4*", + "4*4*4*4*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*7*7*7*7*7*7*7*7*7*7*E&'#t@s@'#e$# '#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#=#!.s 8#M } f g {.^.3.c.d.e.e.d.d.d.c.3.y ].{.N w 1 } < K.t L ]#~# #d#'#'#'#'#'#'#'#'#Z%7*7*7*7*7*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*4*4*4*4*4*4*", + "5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*Q@p#b@'#l.7*S+'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#f@0#9#L _ u [ 1 f g #.{.].^.^.^.^.^.].{.#.g w 2 | [ c K.t b.g#N@u@'#'#'#'#'#'#'#'#'## 7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*5*", + "5*5*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*E&'#6&2## E&8*E&'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#-#6#^#b.8#u d } 1 f w x g N N g g g w f 2 | e d : @.t b.0 }#'#'#'#'#'#'#'#'#'## S+E&8*8*8*8*8*8*8*8*8*8*8*8*8*8*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*6*5*5*5*5*", + "6*6*6*6*6*6*6*6*6*6*6*6*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*S+'#c#'#W$9*9*9*y.'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#j@:#|#]#J.a b : < e v | 1 2 2 2 1 1 | v e d M u A.~.b.0 6#_#'#'#'#'#'#'#'## S+!&8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*6*6*6*6*6*6*6*6*6*6*6*6*6*6*", + "7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*J&'#1#'#W+9*9*9*9*W$'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#W@=#:#]#s S._ b : c d [ e e e [ [ d c M u @.8#S.J.0 }#=#*#(#'#'#`@>#j@'#<&9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*7*6*", + "7*7*7*7*7*7*7*7*7*7*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*0*0*0*0*0*W+'#'#y.0*0*0*0*0*0*@ '#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#I@/#5#_#0#0 b.a 8#@.b K.: : : : u K.b A.8#a ( s g#}#~#/#-#N ++=+N.L [email protected]*0*0*0*0*0*0*0*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*7*7*7*7*7*7*7*7*7*7*7*7*", + "8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*Q&'#'#@ Q&0*0*0*0*0*0*7&'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#K@Y@(#5#|#9#7#J.S.a t 8#_ _ _ 8#t ~.S.( L 7#]#h#5#>#W@_#:+n F+y *#@ Q&0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*8*", + "8*8*8*8*8*8*8*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*a*a*a*a*a*a*a*a*a*a*a*;%'#'#;%a*a*a*a*a*a*a*a*H.'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#J@X@Z@-#/#:#0#g#7#L J.( ( ( ( ( J.L 2.0 9#6#:#;#W@`@L@~.=+e+P.[ v@;%a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*8*8*8*8*8*8*8*8*8*", + "9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*U&'#'#j#a*a*a*a*a*a*a*b*b*b*'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#V@K@X@ #W@*#_#}#^#9#!.0 0 0 0 !.9#^#}#{#5#># #O@M@K@n.%.n+n.s c*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*9*", + "9*9*9*9*9*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*;%'#H.b*b*b*b*b*b*b*b*b*b*b*9&'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#t@V@x@M@L@X@Y@;#=#:#h#|#|#}#|#h#{#=#*#>#`@L@M@v@j@]#:+-.<.f d*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*9*9*9*9*9*9*9*", + "0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*H.H.e*e*e*e*e*e*e*e*e*e*e*e*e*d&'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#8&I@j@w@v@z@L@`@Y@(#;#*#*#;#/#>#W@ #K@x@w@V@I@b@'#=#;+)+J.f*e*e*e*e*e*e*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*0*", + "0*0*0*0*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*`&e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*d&'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#E#q#I@h@i@j@l@y@M@O@X@X@X@L@M@x@l@j@h@J@q#E#1#'#'#'#+ g*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*0*0*0*0*0*0*", + "a*a*a*a*a*a*a*a*a*a*a*a*a*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*d&'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#2#E#u@q#H@e@e@I@i@V@V@j@h@I@H@H@q#u@s@c#'#'#'#'#Y.+*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*a*a*a*a*a*a*a*a*a*a*a*a*a*a*a*", + "a*a*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*Y.'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#b#p#6&s@E#s@l#s@s@s@E#E#E#l#p#1#'#'#'#'#'#u#h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*b*a*a*a*a*", + "b*b*b*b*b*b*b*b*b*b*b*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*z#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#1#1#1#'#'#'#'#'#'#'#'#'#'#'#'#'#+ 1%i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*b*b*b*b*b*b*b*b*b*b*b*b*b*", + "b*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*j*j*j*j*j*j*j*j*j*j*j*j*j*l&+ '#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#-@j*j*j*j*j*j*j*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*e*b*b*b*", + "e*e*e*e*e*e*e*e*e*e*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*h%+ '#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#k*l*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*e*e*e*e*e*e*e*e*e*e*e*e*", + "h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*o%. '#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#n*o*t$p*q*r*s*m*m*m*m*m*m*m*m*m*m*m*m*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*h*e*", + "h*h*h*h*h*h*h*h*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*t*u**&a%v*'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#'#{%w*x*t$y*h$c$c$c$h$h$z*A*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*h*h*h*h*h*h*h*h*h*h*", + "i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*C*q*D*D*D*E*P$,%{&F*'#'#'#'#'#'#'#'#'#'#'#'#'#G*H*I*g$J*I%p*y*c$K*L*7$7$7$r%r%r%7$M*B*B*B*B*B*B*B*B*B*B*B*B*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*i*", + "i*i*i*i*i*i*i*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*N*N*N*N*O*<$K$P*Q*Q$R*S*7$L*r%b%3%T*;$;$;$)@)@)@;$G%t&G%U*V*W*X*%%D*t$h$K*7$S*Y*Y*i%R*b%b%b%b%Z*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*i*i*i*i*i*i*i*i*i*", + "j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*`* =.=+=v$@=@=v$#=,$$=8$Q$<$%=8$&=N*N*N*N*N*N*N*N*N*t&*===-= &Z$q*y%L*S*i%b%3%<%;=^%>=,=&%>='=N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*j*", + "j*j*j*j*j*j*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*)=)=)=)=)=)=)=)=)=!=~=c${=]=R$9$^=/=+%^=9$v$(=+=_=+=:=)=)=)=)=)=)=)=)=)=)=<=[=Y$s$ &J*q*h$r%b%<%^%'%}=|=1=2=3=t$4=N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*j*j*j*j*j*j*j*j*", + "m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=5=6=,%c$^%7=]=8=]$z$9=0=a=0=b=b=+%c=d=)=)=)=)=)=)=)=)=)=)=e=f=g=h=s$ &E*p*i=j=k=l=m=n=o=p=q=r=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*m*", + "m*m*m*m*m*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=t=u=*&v=y*i%&%%=1$w=H$x=y=w$'$b$o$[$z=s=s=s=s=s=s=s=s=s=s=s=y&A=B=_%>%k*-=C=D=E=x*F=O#G=y&s=s=s=s=s=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*m*m*m*m*m*m*m*", + "B*B*B*B*B*B*B*B*B*B*B*B*B*B*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=H=I=J=I%y%K*Y*;=;=u$%=K=_=@=z$L=0=#=M=s=s=s=s=s=s=s=s=s=s=s=s=s=<*G=N=H=f$O=t=O#y&s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*B*", + "B*B*B*B*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=e$Q=n*R=S=S=T=U=V=W=X=Y=Z=`= -.-+-@-#-P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*B*B*B*B*B*B*", + "N*N*N*N*N*N*N*N*N*N*N*N*N*)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=9+$-%-&-*-=---n*;->-,-'-)-!-~-{-]-^-^-^-^-^-^-^-^-P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=N*N*N*N*N*N*N*N*N*N*N*N*N*N*N*", + "N*N*N*N*)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-J&p@/-/-(-p@e$e$p@_-E@o@(-:-|*^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=)=N*N*N*N*N*N*", + ")=)=)=)=)=)=)=)=)=)=)=)=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=)=)=)=)=)=)=)=)=)=)=)=)=)=)=", + ")=)=)=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=s=)=)=)=)=)=", + "s=s=s=s=s=s=s=s=s=s=s=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=s=s=s=s=s=s=s=s=s=s=s=s=s=", + "s=s=s=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=P=s=s=s=s=s=", + "P=P=P=P=P=P=P=P=P=P=P=^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-P=P=P=P=P=P=P=P=P=P=P=P=P=", + "P=P=P=^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-}-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-[-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-<-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-^-P=P=P=P=P=" + }; + +#endif //IMAGE_H diff --git a/src/importers/Makefile.am b/src/importers/Makefile.am new file mode 100644 index 0000000..28ab637 --- /dev/null +++ b/src/importers/Makefile.am @@ -0,0 +1,15 @@ +## Makefile.am for krecipes + +# this is the program that gets installed. it's name is used for all +# of the other Makefile.am variables + +# set the include path for X, tqt and TDE +INCLUDES = -I$(srcdir)/.. -I$(srcdir)/../backends $(all_includes) + +noinst_LTLIBRARIES=libkrecipesimporters.la +libkrecipesimporters_la_SOURCES = mx2importer.cpp mmfimporter.cpp mxpimporter.cpp nycgenericimporter.cpp recipemlimporter.cpp baseimporter.cpp kreimporter.cpp rezkonvimporter.cpp kredbimporter.cpp +libkrecipesimporters_la_METASOURCES=AUTO + +#the library search path. +libkrecipesimporters_la_LDFLAGS = $(KDE_RPATH) $(all_libraries) +noinst_HEADERS = kreimporter.h diff --git a/src/importers/baseimporter.cpp b/src/importers/baseimporter.cpp new file mode 100644 index 0000000..b0ecd61 --- /dev/null +++ b/src/importers/baseimporter.cpp @@ -0,0 +1,399 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "baseimporter.h" + +#include <tdeapplication.h> +#include <tdeconfig.h> +#include <kdebug.h> +#include <tdelocale.h> +#include <kprogress.h> +#include <tdemessagebox.h> + +#include <tqvaluevector.h> + +#include "datablocks/recipe.h" +#include "backends/recipedb.h" +#include "datablocks/categorytree.h" +#include "datablocks/unit.h" + +BaseImporter::BaseImporter() : + m_recipe_list( new RecipeList ), + m_cat_structure( 0 ), + file_recipe_count( 0 ) +{ + TDEConfig * config = kapp->config(); + config->setGroup( "Import" ); + + direct = config->readBoolEntry( "DirectImport", false ); +} + +BaseImporter::~BaseImporter() +{ + delete m_recipe_list; + delete m_cat_structure; +} + +void BaseImporter::add( const RecipeList &recipe_list ) +{ + file_recipe_count += recipe_list.count(); + + for ( RecipeList::const_iterator it = recipe_list.begin(); it != recipe_list.end(); ++it ) { + Recipe copy = *it; + copy.recipeID = -1; //make sure an importer didn't give this a value + for ( RatingList::iterator rating_it = copy.ratingList.begin(); rating_it != copy.ratingList.end(); ++rating_it ) { + (*rating_it).id = -1; + } + m_recipe_list->append( copy ); + } + + if ( direct ) { + if ( !m_progress_dialog->wasCancelled() ) + importRecipes( *m_recipe_list, m_database, m_progress_dialog ); + } +} + +void BaseImporter::add( const Recipe &recipe ) +{ + file_recipe_count++; + Recipe copy = recipe; + copy.recipeID = -1; //make sure an importer didn't give this a value + + if ( direct ) { + if ( !m_progress_dialog->wasCancelled() ) { + RecipeList list; + list.append( recipe ); + importRecipes( list, m_database, m_progress_dialog ); + } + } + else + m_recipe_list->append( copy ); +} + +void BaseImporter::parseFiles( const TQStringList &filenames ) +{ + if ( direct ) + m_filenames = filenames; + else { + for ( TQStringList::const_iterator file_it = filenames.begin(); file_it != filenames.end(); ++file_it ) { + file_recipe_count = 0; + parseFile( *file_it ); + processMessages( *file_it ); + } + } +} + +void BaseImporter::import( RecipeDB *db, bool automatic ) +{ + if ( direct ) { + m_database = db; + + m_progress_dialog = new KProgressDialog( kapp->mainWidget(), 0, + i18n( "Importing selected recipes" ), TQString::null, true ); + KProgress *progress = m_progress_dialog->progressBar(); + progress->setPercentageVisible(false); + progress->setTotalSteps( 0 ); + + for ( TQStringList::const_iterator file_it = m_filenames.begin(); file_it != m_filenames.end(); ++file_it ) { + file_recipe_count = 0; + parseFile( *file_it ); + processMessages( *file_it ); + + if ( m_progress_dialog->wasCancelled() ) + break; + } + + importUnitRatios( db ); + delete m_progress_dialog; + } + else { + if ( m_recipe_list->count() == 0 ) + return; + + m_recipe_list->empty(); + //db->blockSignals(true); + + m_progress_dialog = new KProgressDialog( kapp->mainWidget(), 0, + i18n( "Importing selected recipes" ), TQString::null, true ); + KProgress *progress = m_progress_dialog->progressBar(); + progress->setTotalSteps( m_recipe_list->count() ); + progress->setFormat( i18n( "%v/%m Recipes" ) ); + + if ( m_cat_structure ) { + importCategoryStructure( db, m_cat_structure ); + delete m_cat_structure; + m_cat_structure = 0; + } + importRecipes( *m_recipe_list, db, m_progress_dialog ); + importUnitRatios( db ); + + //db->blockSignals(false); + delete m_progress_dialog; m_progress_dialog = 0; + } +} + +void BaseImporter::importIngredient( IngredientData &ing, RecipeDB *db, KProgressDialog *progress_dialog ) +{ + //cache some data we'll need + int max_units_length = db->maxUnitNameLength(); + int max_group_length = db->maxIngGroupNameLength(); + + if ( direct ) { + progress_dialog->progressBar()->advance( 1 ); + kapp->processEvents(); + } + + //create ingredient groups + TQString real_group_name = ing.group.left( max_group_length ); + int new_group_id = db->findExistingIngredientGroupByName(real_group_name); + if ( new_group_id == -1 ) { + db->createNewIngGroup( real_group_name ); + new_group_id = db->lastInsertID(); + } + ing.groupID = new_group_id; + + int new_ing_id = db->findExistingIngredientByName(ing.name); + if ( new_ing_id == -1 && !ing.name.isEmpty() ) + { + db->createNewIngredient( ing.name ); + new_ing_id = db->lastInsertID(); + } + + if ( direct ) { + progress_dialog->progressBar()->advance( 1 ); + kapp->processEvents(); + } + + Unit real_unit( ing.units.name.left( max_units_length ), ing.units.plural.left( max_units_length ) ); + if ( real_unit.name.isEmpty() ) + real_unit.name = real_unit.plural; + else if ( real_unit.plural.isEmpty() ) + real_unit.plural = real_unit.name; + + int new_unit_id = db->findExistingUnitByName(real_unit.name); + if ( new_unit_id == -1 ) { + db->createNewUnit( Unit(real_unit.name, real_unit.plural) ); + new_unit_id = db->lastInsertID(); + } + + if ( direct ) { + progress_dialog->progressBar()->advance( 1 ); + kapp->processEvents(); + } + + if ( ing.prepMethodList.count() > 0 ) { + for ( ElementList::iterator prep_it = ing.prepMethodList.begin(); prep_it != ing.prepMethodList.end(); ++prep_it ) { + int new_prep_id = db->findExistingPrepByName((*prep_it).name); + if ( new_prep_id == -1 ) { + db->createNewPrepMethod((*prep_it).name); + new_prep_id = db->lastInsertID(); + } + (*prep_it).id = new_prep_id; + } + } + + ing.units.id = new_unit_id; + ing.ingredientID = new_ing_id; + + if ( !db->ingredientContainsUnit( new_ing_id, new_unit_id ) ) + db->addUnitToIngredient( new_ing_id, new_unit_id ); +} + +void BaseImporter::importRecipes( RecipeList &selected_recipes, RecipeDB *db, KProgressDialog *progress_dialog ) +{ + // Load Current Settings + TDEConfig *config = kapp->config(); + config->setGroup( "Import" ); + bool overwrite = config->readBoolEntry( "OverwriteExisting", false ); + + RecipeList::iterator recipe_it; RecipeList::iterator recipe_list_end( selected_recipes.end() ); + RecipeList::iterator recipe_it_old = selected_recipes.end(); + for ( recipe_it = selected_recipes.begin(); recipe_it != recipe_list_end; ++recipe_it ) { + if ( !direct ) { + if ( progress_dialog->wasCancelled() ) { + KMessageBox::information( kapp->mainWidget(), i18n( "All recipes up unto this point have been successfully imported." ) ); + //db->blockSignals(false); + return ; + } + } + + if ( recipe_it_old != selected_recipes.end() ) + selected_recipes.remove( recipe_it_old ); + + progress_dialog->setLabel( TQString( i18n( "Importing recipe: %1" ) ).arg( ( *recipe_it ).title ) ); + progress_dialog->progressBar()->advance( 1 ); + kapp->processEvents(); + + //add all recipe items (authors, ingredients, etc. to the database if they aren't already + IngredientList::iterator ing_list_end( ( *recipe_it ).ingList.end() ); + for ( IngredientList::iterator ing_it = ( *recipe_it ).ingList.begin(); ing_it != ing_list_end; ++ing_it ) { + importIngredient( *ing_it, db, progress_dialog ); + + for ( TQValueList<IngredientData>::iterator sub_it = (*ing_it).substitutes.begin(); sub_it != (*ing_it).substitutes.end(); ++sub_it ) { + importIngredient( *sub_it, db, progress_dialog ); + } + } + + ElementList::iterator author_list_end( ( *recipe_it ).authorList.end() ); + for ( ElementList::iterator author_it = ( *recipe_it ).authorList.begin(); author_it != author_list_end; ++author_it ) { + if ( direct ) { + progress_dialog->progressBar()->advance( 1 ); + kapp->processEvents(); + } + + int new_author_id = db->findExistingAuthorByName(( *author_it ).name); + if ( new_author_id == -1 && !( *author_it ).name.isEmpty() ) { + db->createNewAuthor( ( *author_it ).name ); + new_author_id = db->lastInsertID(); + } + + ( *author_it ).id = new_author_id; + } + + ElementList::iterator cat_list_end( ( *recipe_it ).categoryList.end() ); + for ( ElementList::iterator cat_it = ( *recipe_it ).categoryList.begin(); cat_it != cat_list_end; ++cat_it ) { + if ( direct ) { + progress_dialog->progressBar()->advance( 1 ); + kapp->processEvents(); + } + + int new_cat_id = db->findExistingCategoryByName(( *cat_it ).name); + if ( new_cat_id == -1 && !( *cat_it ).name.isEmpty() ) { + db->createNewCategory( ( *cat_it ).name ); + new_cat_id = db->lastInsertID(); + } + + ( *cat_it ).id = new_cat_id; + } + + if ( !(*recipe_it).yield.type.isEmpty() ) { + int new_id = db->findExistingYieldTypeByName((*recipe_it).yield.type); + if ( new_id == -1 ) { + db->createNewYieldType( (*recipe_it).yield.type ); + new_id = db->lastInsertID(); + } + (*recipe_it).yield.type_id = new_id; + } + + RatingList::iterator rating_list_end( ( *recipe_it ).ratingList.end() ); + for ( RatingList::iterator rating_it = ( *recipe_it ).ratingList.begin(); rating_it != rating_list_end; ++rating_it ) { + if ( direct ) { + progress_dialog->progressBar()->advance( 1 ); + kapp->processEvents(); + } + + for ( RatingCriteriaList::iterator rc_it = (*rating_it).ratingCriteriaList.begin(); rc_it != (*rating_it).ratingCriteriaList.end(); ++rc_it ) { + int new_criteria_id = db->findExistingRatingByName(( *rc_it ).name); + if ( new_criteria_id == -1 && !( *rc_it ).name.isEmpty() ) { + db->createNewRating( ( *rc_it ).name ); + new_criteria_id = db->lastInsertID(); + } + + ( *rc_it ).id = new_criteria_id; + } + } + + if ( overwrite ) //overwrite existing + ( *recipe_it ).recipeID = db->findExistingRecipeByName( ( *recipe_it ).title ); + else //rename + ( *recipe_it ).title = db->getUniqueRecipeTitle( ( *recipe_it ).title ); + + if ( direct ) { + progress_dialog->progressBar()->advance( 1 ); + kapp->processEvents(); + } + + //save into the database + db->saveRecipe( &( *recipe_it ) ); + + recipe_it_old = recipe_it; //store to delete once we've got the next recipe + } +} + +void BaseImporter::setCategoryStructure( CategoryTree *cat_structure ) +{ + if ( direct ) { + importCategoryStructure( m_database, cat_structure ); + } + else { + delete m_cat_structure; + m_cat_structure = cat_structure; + } +} + +void BaseImporter::importCategoryStructure( RecipeDB *db, const CategoryTree *categoryTree ) +{ + for ( CategoryTree * child_it = categoryTree->firstChild(); child_it; child_it = child_it->nextSibling() ) { + int new_cat_id = db->findExistingCategoryByName( child_it->category.name ); + if ( new_cat_id == -1 ) { + db->createNewCategory( child_it->category.name, categoryTree->category.id ); + new_cat_id = db->lastInsertID(); + } + + child_it->category.id = new_cat_id; + + importCategoryStructure( db, child_it ); + } +} + +void BaseImporter::setUnitRatioInfo( UnitRatioList &ratioList, UnitList &unitList ) +{ + m_ratioList = ratioList; + m_unitList = unitList; +} + +void BaseImporter::importUnitRatios( RecipeDB *db ) +{ + for ( UnitRatioList::const_iterator it = m_ratioList.begin(); it != m_ratioList.end(); ++it ) { + TQString unitName1, unitName2; + for ( UnitList::const_iterator unit_it = m_unitList.begin(); unit_it != m_unitList.end(); ++unit_it ) { + if ( ( *it ).uID1 == ( *unit_it ).id ) { + unitName1 = ( *unit_it ).name; + if ( !unitName2.isNull() ) + break; + } + else if ( ( *it ).uID2 == ( *unit_it ).id ) { + unitName2 = ( *unit_it ).name; + if ( !unitName1.isNull() ) + break; + } + } + + int unitId1 = db->findExistingUnitByName( unitName1 ); + int unitId2 = db->findExistingUnitByName( unitName2 ); + + //the unit needed for the ratio may not have been added, because the + //recipes chosen did not include the unit + if ( unitId1 != -1 && unitId2 != -1 ) { + UnitRatio ratio; + ratio.uID1 = unitId1; + ratio.uID2 = unitId2; + ratio.ratio = ( *it ).ratio; + db->saveUnitRatio( &ratio ); + } + } +} + +void BaseImporter::processMessages( const TQString &file ) +{ + if ( m_error_msgs.count() > 0 ) { + //<!doc> ensures it is detected as RichText + m_master_error += TQString( i18n( "<!doc>Import of recipes from the file <b>\"%1\"</b> <b>failed</b> due to the following error(s):" ) ).arg( file ); + m_master_error += "<ul><li>" + m_error_msgs.join( "</li><li>" ) + "</li></ul>"; + + m_error_msgs.clear(); + } + else if ( m_warning_msgs.count() > 0 ) { + m_master_warning += TQString( i18n( "The file <b>%1</b> generated the following warning(s):" ) ).arg( file ); + m_master_warning += "<ul><li>" + m_warning_msgs.join( "</li><li>" ) + "</li></ul>"; + + m_warning_msgs.clear(); + } +} diff --git a/src/importers/baseimporter.h b/src/importers/baseimporter.h new file mode 100644 index 0000000..944f50e --- /dev/null +++ b/src/importers/baseimporter.h @@ -0,0 +1,126 @@ +/*************************************************************************** +* Copyright (C) 2003-2005 by * +* Unai Garro <[email protected]> * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef BASEIMPORTER_H +#define BASEIMPORTER_H + +#include <tdelocale.h> + +#include <tqstring.h> +#include <tqstringlist.h> + +#include "datablocks/recipelist.h" +#include "datablocks/elementlist.h" +#include "datablocks/unitratiolist.h" + +class Recipe; +class RecipeDB; +class CategoryTree; +class IngredientData; + +class KProgressDialog; + +/** @brief Subclass this class to create an importer for a specific file type. + * + * Subclasses should take the file name of the file to import in their constructor + * and then parse the file. For every recipe found in the file, a Recipe object should + * be created and added to the importer using the @ref add() function. + * + * @author Jason Kivlighn + */ +class BaseImporter +{ +public: + BaseImporter(); + virtual ~BaseImporter(); + + TQString getMessages() const + { + return m_master_error + m_master_warning; + } + TQString getErrorMsg() const + { + return m_master_error; + } + TQString getWarningMsg() const + { + return m_master_warning; + } + + void parseFiles( const TQStringList &filenames ); + + /** Import all the recipes into the given database. These recipes are the + * recipes added to this class by a subclass using the @ref add() method. + */ + void import( RecipeDB *db, bool automatic = false ); + + RecipeList recipeList() const { return *m_recipe_list; } + void setRecipeList( const RecipeList &list ) { *m_recipe_list = list; } + + const CategoryTree *categoryStructure() const { return m_cat_structure; } + +protected: + virtual void parseFile( const TQString &filename ) = 0; + + void importRecipes( RecipeList &selected_recipes, RecipeDB *db, KProgressDialog *progess_dialog ); + + /** Add a recipe to be imported into the database */ + void add( const Recipe &recipe ); + void add( const RecipeList &recipe_list ); + + void setCategoryStructure( CategoryTree *cat_structure ); + void setUnitRatioInfo( UnitRatioList &ratioList, UnitList &unitList ); + + int totalCount() const + { + return m_recipe_list->count(); + } + int fileRecipeCount() const + { + return file_recipe_count; + } + + void setErrorMsg( const TQString & s ) + { + m_error_msgs.append( s ); + } + void addWarningMsg( const TQString & s ) + { + m_warning_msgs.append( s ); + } + +private: + void importCategoryStructure( RecipeDB *, const CategoryTree * ); + void importUnitRatios( RecipeDB * ); + void importIngredient( IngredientData &ing, RecipeDB *db, KProgressDialog *progress_dialog ); + + void processMessages( const TQString &file ); + + RecipeList *m_recipe_list; + CategoryTree *m_cat_structure; + UnitRatioList m_ratioList; + UnitList m_unitList; + + TQStringList m_warning_msgs; + TQStringList m_error_msgs; + TQString m_master_warning; + TQString m_master_error; + + int file_recipe_count; + bool direct; + + RecipeDB *m_database; + KProgressDialog *m_progress_dialog; + TQStringList m_filenames; +}; + +#endif //BASEIMPORTER_H diff --git a/src/importers/kredbimporter.cpp b/src/importers/kredbimporter.cpp new file mode 100644 index 0000000..ebb4d3c --- /dev/null +++ b/src/importers/kredbimporter.cpp @@ -0,0 +1,67 @@ +/*************************************************************************** +* Copyright (C) 2004 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "kredbimporter.h" + +#include <tdeapplication.h> +#include <tdeconfig.h> +#include <kdebug.h> + +#include "datablocks/recipelist.h" +#include "datablocks/categorytree.h" +#include "backends/recipedb.h" + +KreDBImporter::KreDBImporter( const TQString &_dbType, const TQString &_host, const TQString &_user, const TQString &_pass, int _port ) : BaseImporter(), + dbType( _dbType ), + host( _host ), + user( _user ), + pass( _pass ), + port( _port ) +{} + +KreDBImporter::~KreDBImporter() +{} + +void KreDBImporter::parseFile( const TQString &file ) //this is either a database file or a database table +{ + RecipeDB * database = RecipeDB::createDatabase( dbType, host, user, pass, file, port, file ); //uses 'file' as either table or file name, depending on the database + + if ( database ) { + database->connect( false ); //don't create the database if it fails to connect + + if ( database->ok() ) { + //set the category structure + CategoryTree * tree = new CategoryTree; + database->loadCategories( tree ); + setCategoryStructure( tree ); + + #if 0 + //set unit ratios + UnitRatioList ratioList; + UnitList unitList; + database->loadUnitRatios( &ratioList ); + database->loadUnits( &unitList ); + + setUnitRatioInfo( ratioList, unitList ); + #endif + + //now load recipes + RecipeList recipes; + database->loadRecipes( &recipes, RecipeDB::All ^ RecipeDB::Properties ); + + //now add these recipes to the importer + add( recipes ); + } + else + setErrorMsg( database->err() ); + } + + delete database; +} diff --git a/src/importers/kredbimporter.h b/src/importers/kredbimporter.h new file mode 100644 index 0000000..1afc8e8 --- /dev/null +++ b/src/importers/kredbimporter.h @@ -0,0 +1,38 @@ +/*************************************************************************** +* Copyright (C) 2004 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef KREDBIMPORTER_H +#define KREDBIMPORTER_H + +#include <tqstring.h> + +#include "baseimporter.h" + +/** Class to import recipes from any other Krecipes database backend. + * Note: Though independant of database type, the two databases must have the same structure (i.e. be the same version) + * @author Jason Kivlighn + */ +class KreDBImporter : public BaseImporter +{ +public: + KreDBImporter( const TQString &dbType, const TQString &host = TQString::null, const TQString &user = TQString::null, const TQString &pass = TQString::null, int port = 0 ); + virtual ~KreDBImporter(); + +private: + virtual void parseFile( const TQString &file_or_table ); + + TQString dbType; + TQString host; + TQString user; + TQString pass; + int port; +}; + +#endif //KREDBIMPORTER_H diff --git a/src/importers/kreimporter.cpp b/src/importers/kreimporter.cpp new file mode 100644 index 0000000..2e55589 --- /dev/null +++ b/src/importers/kreimporter.cpp @@ -0,0 +1,309 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Cyril Bosselut ([email protected]) * +* * +* Copyright (C) 2003-2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "kreimporter.h" + +#include <tdelocale.h> +#include <kdebug.h> + +#include <tqfile.h> +#include <tqstringlist.h> +#include <kstandarddirs.h> + +#include "datablocks/recipe.h" +#include "datablocks/categorytree.h" + +KreImporter::KreImporter() : BaseImporter() +{} + +void KreImporter::parseFile( const TQString &filename ) +{ + TQFile * file = 0; + bool unlink = false; + kdDebug() << "loading file: %s" << filename << endl; + + if ( filename.right( 4 ) == ".kre" ) { + file = new TQFile( filename ); + kdDebug() << "file is an archive" << endl; + KTar* kre = new KTar( filename, "application/x-gzip" ); + kre->open( IO_ReadOnly ); + const KArchiveDirectory* dir = kre->directory(); + TQString name; + TQStringList fileList = dir->entries(); + for ( TQStringList::Iterator it = fileList.begin(); it != fileList.end(); ++it ) { + if ( ( *it ).right( 6 ) == ".kreml" ) { + name = *it; + } + } + if ( name.isEmpty() ) { + kdDebug() << "error: Archive doesn't contain a valid Krecipes file" << endl; + setErrorMsg( i18n( "Archive does not contain a valid Krecipes file" ) ); + return ; + } + TQString tmp_dir = locateLocal( "tmp", "" ); + dir->copyTo( tmp_dir ); + file = new TQFile( tmp_dir + name ); + kre->close(); + unlink = true; //remove file after import + } + else { + file = new TQFile( filename ); + } + + if ( file->open( IO_ReadOnly ) ) { + kdDebug() << "file opened" << endl; + TQDomDocument doc; + TQString error; + int line; + int column; + if ( !doc.setContent( file, &error, &line, &column ) ) { + kdDebug() << TQString( "error: \"%1\" at line %2, column %3" ).arg( error ).arg( line ).arg( column ) << endl; + setErrorMsg( TQString( i18n( "\"%1\" at line %2, column %3" ) ).arg( error ).arg( line ).arg( column ) ); + return ; + } + + TQDomElement kreml = doc.documentElement(); + + if ( kreml.tagName() != "krecipes" ) { + setErrorMsg( i18n( "This file does not appear to be a *.kreml file" ) ); + return ; + } + + // TODO Check if there are changes between versions + TQString kreVersion = kreml.attribute( "version" ); + kdDebug() << TQString( i18n( "KreML version %1" ) ).arg( kreVersion ) << endl; + + TQDomNodeList r = kreml.childNodes(); + TQDomElement krecipe; + + for ( unsigned z = 0; z < r.count(); z++ ) { + krecipe = r.item( z ).toElement(); + TQDomNodeList l = krecipe.childNodes(); + if ( krecipe.tagName() == "krecipes-recipe" ) { + Recipe recipe; + for ( unsigned i = 0; i < l.count(); i++ ) { + TQDomElement el = l.item( i ).toElement(); + if ( el.tagName() == "krecipes-description" ) { + readDescription( el.childNodes(), &recipe ); + } + if ( el.tagName() == "krecipes-ingredients" ) { + readIngredients( el.childNodes(), &recipe ); + } + if ( el.tagName() == "krecipes-instructions" ) { + recipe.instructions = el.text().stripWhiteSpace(); + } + if ( el.tagName() == "krecipes-ratings" ) { + readRatings( el.childNodes(), &recipe ); + } + } + add + ( recipe ); + } + else if ( krecipe.tagName() == "krecipes-category-structure" ) { + CategoryTree * tree = new CategoryTree; + readCategoryStructure( l, tree ); + setCategoryStructure( tree ); + } + } + } + if ( unlink ) { + file->remove + (); + } +} + +KreImporter::~KreImporter() +{ +} + +void KreImporter::readCategoryStructure( const TQDomNodeList& l, CategoryTree *tree ) +{ + for ( unsigned i = 0; i < l.count(); i++ ) { + TQDomElement el = l.item( i ).toElement(); + + TQString category = el.attribute( "name" ); + CategoryTree *child_node = tree->add + ( Element( category ) ); + readCategoryStructure( el.childNodes(), child_node ); + } +} + +void KreImporter::readDescription( const TQDomNodeList& l, Recipe *recipe ) +{ + for ( unsigned i = 0; i < l.count(); i++ ) { + TQDomElement el = l.item( i ).toElement(); + if ( el.tagName() == "title" ) { + recipe->title = el.text(); + kdDebug() << "Found title: " << recipe->title << endl; + } + else if ( el.tagName() == "author" ) { + kdDebug() << "Found author: " << el.text() << endl; + recipe->authorList.append( Element( el.text() ) ); + } + else if ( el.tagName() == "serving" ) { //### Keep for < 0.9 compatibility + recipe->yield.amount = el.text().toInt(); + } + else if ( el.tagName() == "yield" ) { + TQDomNodeList yield_children = el.childNodes(); + for ( unsigned j = 0; j < yield_children.count(); j++ ) { + TQDomElement y = yield_children.item( j ).toElement(); + if ( y.tagName() == "amount" ) + readAmount(y,recipe->yield.amount,recipe->yield.amount_offset); + else if ( y.tagName() == "type" ) + recipe->yield.type = y.text(); + } + } + else if ( el.tagName() == "preparation-time" ) { + recipe->prepTime = TQTime::fromString( el.text() ); + } + else if ( el.tagName() == "category" ) { + TQDomNodeList categories = el.childNodes(); + for ( unsigned j = 0; j < categories.count(); j++ ) { + TQDomElement c = categories.item( j ).toElement(); + if ( c.tagName() == "cat" ) { + kdDebug() << "Found category: " << TQString( c.text() ).stripWhiteSpace() << endl; + recipe->categoryList.append( Element( TQString( c.text() ).stripWhiteSpace() ) ); + } + } + } + else if ( el.tagName() == "pictures" ) { + if ( el.hasChildNodes() ) { + TQDomNodeList pictures = el.childNodes(); + for ( unsigned j = 0; j < pictures.count(); j++ ) { + TQDomElement pic = pictures.item( j ).toElement(); + TQCString decodedPic; + if ( pic.tagName() == "pic" ) + kdDebug() << "Found photo" << endl; + TQPixmap pix; + KCodecs::base64Decode( TQCString( pic.text().latin1() ), decodedPic ); + int len = decodedPic.size(); + TQByteArray picData( len ); + memcpy( picData.data(), decodedPic.data(), len ); + bool ok = pix.loadFromData( picData, "JPEG" ); + if ( ok ) { + recipe->photo = pix; + } + } + } + } + } +} + +void KreImporter::readIngredients( const TQDomNodeList& l, Recipe *recipe, const TQString &header, Ingredient *ing ) +{ + for ( unsigned i = 0; i < l.count(); i++ ) { + TQDomElement el = l.item( i ).toElement(); + if ( el.tagName() == "ingredient" ) { + TQDomNodeList ingredient = el.childNodes(); + Ingredient new_ing; + for ( unsigned j = 0; j < ingredient.count(); j++ ) { + TQDomElement ing = ingredient.item( j ).toElement(); + if ( ing.tagName() == "name" ) { + new_ing.name = TQString( ing.text() ).stripWhiteSpace(); + kdDebug() << "Found ingredient: " << new_ing.name << endl; + } + else if ( ing.tagName() == "amount" ) { + readAmount(ing,new_ing.amount,new_ing.amount_offset); + } + else if ( ing.tagName() == "unit" ) { + new_ing.units = Unit( ing.text().stripWhiteSpace(), new_ing.amount+new_ing.amount_offset ); + } + else if ( ing.tagName() == "prep" ) { + new_ing.prepMethodList = ElementList::split(",",TQString( ing.text() ).stripWhiteSpace()); + } + else if ( ing.tagName() == "substitutes" ) { + readIngredients(ing.childNodes(), recipe, header, &new_ing); + } + } + new_ing.group = header; + + if ( !ing ) + recipe->ingList.append( new_ing ); + else + ing->substitutes.append( new_ing ); + } + else if ( el.tagName() == "ingredient-group" ) { + readIngredients( el.childNodes(), recipe, el.attribute( "name" ) ); + } + } +} + +void KreImporter::readAmount( const TQDomElement& amountEl, double &amount, double &amount_offset ) +{ + TQDomNodeList children = amountEl.childNodes(); + + double min = 0,max = 0; + for ( unsigned i = 0; i < children.count(); i++ ) { + TQDomElement child = children.item( i ).toElement(); + if ( child.tagName() == "min" ) { + min = ( TQString( child.text() ).stripWhiteSpace() ).toDouble(); + } + else if ( child.tagName() == "max" ) + max = ( TQString( child.text() ).stripWhiteSpace() ).toDouble(); + else if ( child.tagName().isEmpty() ) + min = ( TQString( amountEl.text() ).stripWhiteSpace() ).toDouble(); + } + + amount = min; + if ( max > 0 ) + amount_offset = max-min; +} + +void KreImporter::readRatings( const TQDomNodeList& l, Recipe *recipe ) +{ + for ( unsigned i = 0; i < l.count(); i++ ) { + TQDomElement child = l.item( i ).toElement(); + if ( child.tagName() == "rating" ) { + Rating r; + + TQDomNodeList ratingChildren = child.childNodes(); + for ( unsigned j = 0; j < ratingChildren.count(); j++ ) { + TQDomElement ratingChild = ratingChildren.item( j ).toElement(); + if ( ratingChild.tagName() == "comment" ) { + r.comment = ratingChild.text(); + } + else if ( ratingChild.tagName() == "rater" ) { + r.rater = ratingChild.text(); + } + else if ( ratingChild.tagName() == "criterion" ) { + readCriterion(ratingChild.childNodes(),r.ratingCriteriaList); + } + } + recipe->ratingList.append(r); + } + } +} + +void KreImporter::readCriterion( const TQDomNodeList& l, RatingCriteriaList &rc_list ) +{ + for ( unsigned i = 0; i < l.count(); i++ ) { + TQDomElement child = l.item( i ).toElement(); + + if ( child.tagName() == "criteria" ) { + RatingCriteria rc; + + TQDomNodeList criteriaChildren = child.childNodes(); + for ( unsigned j = 0; j < criteriaChildren.count(); j++ ) { + TQDomElement criteriaChild = criteriaChildren.item( j ).toElement(); + + if ( criteriaChild.tagName() == "name" ) { + rc.name = criteriaChild.text(); + } + else if ( criteriaChild.tagName() == "stars" ) { + rc.stars = criteriaChild.text().toDouble(); + } + } + rc_list.append(rc); + } + } +} diff --git a/src/importers/kreimporter.h b/src/importers/kreimporter.h new file mode 100644 index 0000000..af774d2 --- /dev/null +++ b/src/importers/kreimporter.h @@ -0,0 +1,54 @@ +/*************************************************************************** +* Copyright (C) 2003-2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef KREIMPORTER_H +#define KREIMPORTER_H + +#include <tdelocale.h> +#include <kstandarddirs.h> +#include <kmdcodec.h> +#include <ktar.h> +#include <tdetempfile.h> + +#include <tqfile.h> +#include <tqstringlist.h> +#include <tqdom.h> + +#include "baseimporter.h" + +#include "datablocks/recipe.h" + +class Recipe; +class CategoryTree; + +/** +Import for Krecipes native file format (.kre, .kreml) + +@author Cyril Bosselut, Jason Kivlighn +*/ +class KreImporter: public BaseImporter +{ +public: + KreImporter(); + virtual ~KreImporter(); + +private: + void parseFile( const TQString& filename ); + +private: + void readCategoryStructure( const TQDomNodeList& l, CategoryTree *tree ); + void readDescription( const TQDomNodeList& l, Recipe* ); + void readIngredients( const TQDomNodeList& l, Recipe*, const TQString &header = TQString::null, Ingredient *ing = 0 ); + void readAmount( const TQDomElement& amount1, double &amount2, double &amount_offset ); + void readRatings( const TQDomNodeList&, Recipe * ); + void readCriterion( const TQDomNodeList&, RatingCriteriaList &r ); +}; + +#endif diff --git a/src/importers/mmfimporter.cpp b/src/importers/mmfimporter.cpp new file mode 100644 index 0000000..5487d2b --- /dev/null +++ b/src/importers/mmfimporter.cpp @@ -0,0 +1,336 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "mmfimporter.h" + +#include <tdeapplication.h> +#include <tdelocale.h> +#include <kdebug.h> + +#include <tqfile.h> +#include <tqregexp.h> +#include <tqtextstream.h> +#include <tqstringlist.h> + +#include "datablocks/mixednumber.h" +#include "datablocks/recipe.h" +#include "mmdata.h" + +//TODO: pre-parse file and try to correct alignment errors in ingredients? + +MMFImporter::MMFImporter() : BaseImporter() +{} + +MMFImporter::~MMFImporter() +{} + +void MMFImporter::parseFile( const TQString &file ) +{ + resetVars(); + + TQFile input( file ); + + if ( input.open( IO_ReadOnly ) ) { + TQTextStream stream( &input ); + stream.skipWhiteSpace(); + + TQString line; + while ( !stream.atEnd() ) { + line = stream.readLine(); + + if ( line.startsWith( "MMMMM" ) ) { + version = VersionMMMMM; + importMMF( stream ); + } + else if ( line.contains( "Recipe Extracted from Meal-Master (tm) Database" ) ) { + version = FromDatabase; + importMMF( stream ); + } + else if ( line.startsWith( "-----" ) ) { + version = VersionNormal; + importMMF( stream ); + } + else if ( line.startsWith( "MM" ) ) { + version = VersionBB; + ( void ) stream.readLine(); + importMMF( stream ); + } + + stream.skipWhiteSpace(); + } + + if ( fileRecipeCount() == 0 ) + addWarningMsg( i18n( "No recipes found in this file." ) ); + } + else + setErrorMsg( i18n( "Unable to open file." ) ); +} + +void MMFImporter::importMMF( TQTextStream &stream ) +{ + kapp->processEvents(); //don't want the user to think its frozen... especially for files with thousands of recipes + + TQString current; + + //===============FIXED FORMAT================// + //line 1: title + //line 2: categories (comma or space separated) + //line 3: yield (number followed by label) + + //title + stream.skipWhiteSpace(); + current = stream.readLine(); + m_title = current.mid( current.find( ":" ) + 1, current.length() ).stripWhiteSpace(); + kdDebug() << "Found title: " << m_title << endl; + + //categories + stream.skipWhiteSpace(); + current = stream.readLine().stripWhiteSpace(); + const char separator = ( version == FromDatabase ) ? ' ' : ','; + TQStringList categories = TQStringList::split( separator, current.mid( current.find( ":" ) + 1, current.length() ) ); + for ( TQStringList::const_iterator it = categories.begin(); it != categories.end(); ++it ) { + Element new_cat; + new_cat.name = TQString( *it ).stripWhiteSpace(); + kdDebug() << "Found category: " << new_cat.name << endl; + m_categories.append( new_cat ); + } + + //servings + stream.skipWhiteSpace(); + current = stream.readLine().stripWhiteSpace(); + if ( current.startsWith( "Yield:" ) ) { + //get the number between the ":" and the next space after it + m_servings = current.mid( current.find( ":" ) + 1, + current.find( " ", current.find( ":" ) + 2 ) - current.find( ":" ) ).toInt(); + kdDebug() << "Found yield: " << m_servings << endl; + } + else if ( current.startsWith( "Servings:" ) ) //from database version + { + m_servings = current.mid( current.find( ":" ) + 1, current.length() ).toInt(); + kdDebug() << "Found servings: " << m_servings << endl; + } + + //=======================VARIABLE FORMAT===================// + //read lines until ending is found + //each line is either an ingredient, ingredient header, or instruction + bool instruction_found = false; + bool is_sub = false; + + ( void ) stream.readLine(); + current = stream.readLine(); + while ( current.stripWhiteSpace() != "MMMMM" && + current.stripWhiteSpace() != "-----" && + current.stripWhiteSpace() != "-----------------------------------------------------------------------------" && + !stream.atEnd() ) { + bool col_one_used = loadIngredientLine( current.left( 41 ), m_left_col_ing, is_sub ); + if ( col_one_used ) //only check for second column if there is an ingredient in the first column + loadIngredientLine( current.mid( 41, current.length() ), m_right_col_ing, is_sub ); + + if ( instruction_found && col_one_used ) { + addWarningMsg( TQString( i18n( "While loading recipe <b>%1</b> " + "an ingredient line was found after the directions. " + "While this is valid, it most commonly indicates an incorrectly " + "formatted recipe." ) ).arg( m_title ) ); + } + + if ( !col_one_used && + !loadIngredientHeader( current.stripWhiteSpace() ) ) { + if ( !current.stripWhiteSpace().isEmpty() ) + instruction_found = true; + m_instructions += current.stripWhiteSpace() + "\n"; + //kdDebug()<<"Found instruction line: "<<current.stripWhiteSpace()<<endl; + } + + current = stream.readLine(); + } + m_instructions = m_instructions.stripWhiteSpace(); + //kdDebug()<<"Found instructions: "<<m_instructions<<endl; + + putDataInRecipe(); +} + +bool MMFImporter::loadIngredientLine( const TQString &string, IngredientList &list, bool &is_sub ) +{ + //just ignore an empty line + if ( string.stripWhiteSpace().isEmpty() ) + return false; + + Ingredient new_ingredient; + new_ingredient.amount = 0; //amount not required, so give default of 0 + + if ( string.at( 11 ) == "-" && string.mid( 0, 11 ).stripWhiteSpace().isEmpty() && !list.isEmpty() ) //continuation of previous ingredient + { + //kdDebug()<<"Appending to last ingredient in column: "<<string.stripWhiteSpace().mid(1,string.length())<<endl; + ( *list.at( list.count() - 1 ) ).name += " " + string.stripWhiteSpace().mid( 1, string.length() ); + TQString name = ( *list.at( list.count() - 1 ) ).name; + + if ( name.endsWith(", or") ) { + ( *list.at( list.count() - 1 ) ).name = name.left(name.length()-4); + is_sub = true; + } + else + is_sub = false; + + return true; + } + + //amount + if ( !string.mid( 0, 7 ).stripWhiteSpace().isEmpty() ) { + bool ok; + MixedNumber amount = MixedNumber::fromString( string.mid( 0, 7 ).stripWhiteSpace(), &ok, false ); + if ( !ok ) + return false; + else + new_ingredient.amount = amount.toDouble(); + } + + //amount/unit separator + if ( string[ 7 ] != ' ' ) + return false; + + //unit + if ( !string.mid( 8, 2 ).stripWhiteSpace().isEmpty() ) { + bool is_unit = false; + TQString unit( string.mid( 8, 2 ).stripWhiteSpace() ); + for ( int i = 0; unit_info[ i ].short_form; i++ ) { + if ( unit_info[ i ].short_form == unit ) { + is_unit = true; + if ( new_ingredient.amount <= 1 ) + unit = unit_info[ i ].expanded_form; + else + unit = unit_info[ i ].plural_expanded_form; + + break; + } + } + if ( !is_unit ) { /*This gives too many false warnings... + addWarningMsg( TQString(i18n("Unit \"%1\" not recognized. " + "Used in the context of \"%2\". If this shouldn't be an ingredient line (i.e. is part of the instructions), " + "then you can safely ignore this warning, and the recipe will be correctly imported.")).arg(unit).arg(string.stripWhiteSpace()) );*/ + return false; + } + + if ( int(new_ingredient.amount) > 1 ) + new_ingredient.units.plural = unit; + else + new_ingredient.units.name = unit; + } + + //unit/name separator + if ( string[ 10 ] != ' ' || string[ 11 ] == ' ' ) + return false; + + //name and preparation method + new_ingredient.name = string.mid( 11, 41 ).stripWhiteSpace(); + + //put in the header... it there is no header, current_header will be TQString::null + new_ingredient.group = current_header; + + bool last_is_sub = is_sub; + if ( new_ingredient.name.endsWith(", or") ) { + new_ingredient.name = new_ingredient.name.left(new_ingredient.name.length()-4); + is_sub = true; + } + else + is_sub = false; + + if ( last_is_sub ) + ( *list.at( list.count() - 1 ) ).substitutes.append(new_ingredient); + else + list.append( new_ingredient ); + + //if we made it this far it is an ingredient line + return true; +} + +bool MMFImporter::loadIngredientHeader( const TQString &string ) +{ + if ( ( string.startsWith( "-----" ) || string.startsWith( "MMMMM" ) ) && + string.length() >= 40 && + ( ( string.at( string.length() / 2 ) != "-" ) || + ( string.at( string.length() / 2 + 1 ) != "-" ) || + ( string.at( string.length() / 2 - 1 ) != "-" ) ) ) { + TQString header( string.stripWhiteSpace() ); + + //get only the header name + header.remove( TQRegExp( "^MMMMM" ) ); + header.remove( TQRegExp( "^-*" ) ).remove( TQRegExp( "-*$" ) ); + + kdDebug() << "found ingredient header: " << header << endl; + + //merge all columns before appending to full ingredient list to maintain the ingredient order + for ( IngredientList::iterator ing_it = m_left_col_ing.begin(); ing_it != m_left_col_ing.end(); ++ing_it ) { + m_all_ing.append( *ing_it ); + } + m_left_col_ing.empty(); + + for ( IngredientList::iterator ing_it = m_right_col_ing.begin(); ing_it != m_right_col_ing.end(); ++ing_it ) { + m_all_ing.append( *ing_it ); + } + m_right_col_ing.empty(); + + current_header = header; + return true; + } + else + return false; +} + +void MMFImporter::putDataInRecipe() +{ + for ( IngredientList::const_iterator ing_it = m_left_col_ing.begin(); ing_it != m_left_col_ing.end(); ++ing_it ) + m_all_ing.append( *ing_it ); + for ( IngredientList::const_iterator ing_it = m_right_col_ing.begin(); ing_it != m_right_col_ing.end(); ++ing_it ) + m_all_ing.append( *ing_it ); + + for ( IngredientList::iterator ing_it = m_all_ing.begin(); ing_it != m_all_ing.end(); ++ing_it ) { + TQString name_and_prep = ( *ing_it ).name; + int separator_index = name_and_prep.find( TQRegExp( "(;|,)" ) ); + if ( separator_index != -1 ) { + ( *ing_it ).name = name_and_prep.mid( 0, separator_index ).stripWhiteSpace(); + ( *ing_it ).prepMethodList = ElementList::split(",",name_and_prep.mid( separator_index + 1, name_and_prep.length() ).stripWhiteSpace() ); + } + } + + //create the recipe + Recipe new_recipe; + new_recipe.yield.amount = m_servings; + new_recipe.yield.type = i18n("servings"); + new_recipe.title = m_title; + new_recipe.instructions = m_instructions; + new_recipe.ingList = m_all_ing; + new_recipe.categoryList = m_categories; + new_recipe.authorList = m_authors; + new_recipe.recipeID = -1; + + //put it in the recipe list + add + ( new_recipe ); + + //reset for the next recipe to use these variables + resetVars(); +} + +void MMFImporter::resetVars() +{ + m_left_col_ing.empty(); + m_right_col_ing.empty(); + m_all_ing.empty(); + m_authors.clear(); + m_categories.clear(); + + m_servings = 0; + + m_title = TQString::null; + m_instructions = TQString::null; + + current_header = TQString::null; +} + diff --git a/src/importers/mmfimporter.h b/src/importers/mmfimporter.h new file mode 100644 index 0000000..8e7d0d2 --- /dev/null +++ b/src/importers/mmfimporter.h @@ -0,0 +1,65 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef MMFIMPORTER_H +#define MMFIMPORTER_H + +#include <tqstring.h> + +#include "baseimporter.h" +#include "datablocks/ingredientlist.h" +#include "datablocks/elementlist.h" + +/** Class to import Meal-Master's MMF (Meal-Master Format) file format. + * @author Jason Kivlighn + */ +class MMFImporter : public BaseImporter +{ +public: + MMFImporter(); + virtual ~MMFImporter(); + +private: + enum FormatVersion { FromDatabase, VersionMMMMM, VersionBB, VersionNormal }; + + virtual void parseFile( const TQString &filename ); + + void importMMF( TQTextStream &stream ); + + /** Parses the line and save it if the line is a valid ingredient and return true. + * Returns false if not an ingredient. + */ + bool loadIngredientLine( const TQString &, IngredientList &, bool &is_sub ); + + /** Parses the line and save it if the line is a valid ingredient header and return true. + * Returns false if not an ingredient header. + */ + bool loadIngredientHeader( const TQString & ); + + void resetVars(); + void putDataInRecipe(); + + int m_servings; + + TQString m_instructions; + TQString m_title; + + ElementList m_authors; + ElementList m_categories; + IngredientList m_left_col_ing; + IngredientList m_right_col_ing; + IngredientList m_all_ing; + + FormatVersion version; + + TQString current_header; +}; + +#endif //MMFIMPORTER_H diff --git a/src/importers/mx2importer.cpp b/src/importers/mx2importer.cpp new file mode 100644 index 0000000..75f75a7 --- /dev/null +++ b/src/importers/mx2importer.cpp @@ -0,0 +1,186 @@ +/* +Copyright (C) 2003 Richard L�rk�ng +Copyright (C) 2003 Jason Kivlighn + +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; if not, write to the Free Software +Foundation, Inc., 51 Franklin Steet, Fifth Floor, Boston, MA 02111-1307 USA +*/ + +#include "mx2importer.h" + +#include <tdelocale.h> +#include <kdebug.h> + +#include <tqfile.h> +#include <tqstringlist.h> +#include <tqtextstream.h> +#include <tqdatetime.h> + +#include "datablocks/recipe.h" + + +MX2Importer::MX2Importer() +{} + +void MX2Importer::parseFile( const TQString& filename ) +{ + TQFile file( filename ); + kdDebug() << "loading file: " << filename << endl; + if ( file.open( IO_ReadOnly ) ) { + kdDebug() << "file opened" << endl; + TQDomDocument doc; + + //hopefully a temporary hack, since MasterCook creates invalid xml declarations + TQTextStream stream( &file ); + TQString all_data = stream.read(); + if ( all_data.startsWith( "<?xml" ) ) + all_data.remove( 0, all_data.find( "?>" ) + 2 ); + + TQString error; + int line; + int column; + if ( !doc.setContent( all_data, &error, &line, &column ) ) { + kdDebug() << TQString( "error: \"%1\" at line %2, column %3" ).arg( error ).arg( line ).arg( column ) << endl; + setErrorMsg( TQString( i18n( "\"%1\" at line %2, column %3. This may not be a *.mx2 file." ) ).arg( error ).arg( line ).arg( column ) ); + return ; + } + + TQDomElement mx2 = doc.documentElement(); + + // TODO Check if there are changes between versions + if ( mx2.tagName() != "mx2" /*|| mx2.attribute("source") != "MasterCook 5.0"*/ ) { + setErrorMsg( i18n( "This file does not appear to be a *.mx2 file" ) ); + return ; + } + + TQDomNodeList l = mx2.childNodes(); + + for ( unsigned i = 0; i < l.count(); i++ ) { + TQDomElement el = l.item( i ).toElement(); + + if ( el.tagName() == "RcpE" ) { + Recipe recipe; + recipe.title = el.attribute( "name" ); + + Element author( el.attribute( "author" ) ); + recipe.authorList.append( author ); + + readRecipe( el.childNodes(), &recipe ); + add + ( recipe ); + } + } + } + else + setErrorMsg( i18n( "Unable to open file." ) ); +} + +MX2Importer::~MX2Importer() +{ +} + +void MX2Importer::readRecipe( const TQDomNodeList& l, Recipe *recipe ) +{ + for ( unsigned i = 0; i < l.count(); i++ ) { + TQDomElement el = l.item( i ).toElement(); + + TQString tagName = el.tagName(); + if ( tagName == "Serv" ) { + recipe->yield.amount = el.attribute( "qty" ).toInt(); + recipe->yield.type = i18n("servings"); + } + else if ( tagName == "PrpT" ) + recipe->prepTime = TQTime::fromString( el.attribute( "elapsed" ) ); + else if ( tagName == "CatS" ) { + TQDomNodeList categories = el.childNodes(); + for ( unsigned j = 0; j < categories.count(); j++ ) { + TQDomElement c = categories.item( j ).toElement(); + if ( c.tagName() == "CatT" ) { + if ( c.text().length() > 0 ) { + Element cat( c.text().stripWhiteSpace() ); + recipe->categoryList.append( cat ); + } + } + } + } + else if ( tagName == "IngR" ) { + Ingredient new_ing( el.attribute( "name" ), + el.attribute( "qty" ).toDouble(), + Unit( el.attribute( "unit" ), el.attribute( "qty" ).toDouble() ) ); + if ( el.hasChildNodes() ) { + TQDomNodeList iChilds = el.childNodes(); + for ( unsigned j = 0; j < iChilds.count(); j++ ) { + TQDomElement iChild = iChilds.item( j ).toElement(); + if ( iChild.tagName() == "IPrp" ) + new_ing.prepMethodList.append( Element(iChild.text().stripWhiteSpace()) ); + else if ( iChild.tagName() == "INtI" ) + ; // TODO: What does it mean?... ingredient nutrient info? + } + } + recipe->ingList.append( new_ing ); + } + else if ( tagName == "DirS" ) { + TQStringList directions; + TQDomNodeList dirs = el.childNodes(); + for ( unsigned j = 0; j < dirs.count(); j++ ) { + TQDomElement dir = dirs.item( j ).toElement(); + if ( dir.tagName() == "DirT" ) + directions.append( dir.text().stripWhiteSpace() ); + } + TQString directionsText; + + // TODO This is copied from RecipeML, maybe a TQStringList + // for directions in Recipe instead? + if ( directions.count() > 1 ) { + for ( unsigned i = 1; i <= directions.count(); i++ ) { + if ( i != 1 ) { + directionsText += "\n\n"; + } + + TQString sWith = TQString( "%1. " ).arg( i ); + TQString text = directions[ i - 1 ]; + if ( !text.stripWhiteSpace().startsWith( sWith ) ) + directionsText += sWith; + directionsText += text; + } + } + else + directionsText = directions[ 0 ]; + + recipe->instructions = directionsText; + } + else if ( tagName == "SrvI" ) { + // Don't know what to do with it, for now add it to directions + // btw lets hope this is read after the directions + recipe->instructions += "\n\n" + el.text().stripWhiteSpace(); + } + else if ( tagName == "Note" ) { + // Don't know what to do with it, for now add it to directions + // btw lets hope this is read after the directions + recipe->instructions += "\n\n" + el.text().stripWhiteSpace(); + } + else if ( tagName == "Nutr" ) { + //example: <Nutr>Per Serving (excluding unknown items): 51 Calories; 6g Fat (99.5% calories from fat); trace Protein; trace Carbohydrate; 0g Dietary Fiber; 16mg Cholesterol; 137mg Sodium. Exchanges: 1 Fat.</Nutr> + // Don't know what to do with it, for now add it to directions + // btw lets hope this is read after the directions + recipe->instructions += "\n\n" + el.text().stripWhiteSpace(); + } + /* tags to check for (example follows: + <Srce>SARA'S SECRETS with Sara Moulton - (Show # SS-1B43) - from the TV FOOD NETWORK</Srce> + <AltS label="Formatted for MC7" source="07-11-2003 by Joe Comiskey - Mad's Recipe Emporium"/> + */ + // TODO Have i missed some tag? + } +} + diff --git a/src/importers/mx2importer.h b/src/importers/mx2importer.h new file mode 100644 index 0000000..df298dd --- /dev/null +++ b/src/importers/mx2importer.h @@ -0,0 +1,46 @@ +/* +Copyright (C) 2003 Richard L�rk�ng +Copyright (C) 2003 Jason Kivlighn + +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; if not, write to the Free Software +Foundation, Inc., 51 Franklin Steet, Fifth Floor, Boston, MA 02111-1307 USA +*/ + +#ifndef MX2IMPORTER_H +#define MX2IMPORTER_H + +#include "baseimporter.h" + +#include <tqdom.h> + +class Recipe; + +/** Class to import Mastercook's MX2 file format. This is an XML-based file + * format used since version 5 of Mastercook. + * @author Jason Kivlighn + */ +class MX2Importer : public BaseImporter +{ +public: + MX2Importer(); + virtual ~MX2Importer(); + +protected: + void parseFile( const TQString& filename ); + +private: + void readRecipe( const TQDomNodeList& l, Recipe* ); +} ; + +#endif //MX2IMPORTER_H diff --git a/src/importers/mxpimporter.cpp b/src/importers/mxpimporter.cpp new file mode 100644 index 0000000..20fbf0f --- /dev/null +++ b/src/importers/mxpimporter.cpp @@ -0,0 +1,382 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "mxpimporter.h" + +#include <tqfile.h> +#include <tqtextstream.h> +#include <tqstringlist.h> +#include <tqdatetime.h> + +#include <tdeapplication.h> +#include <tdelocale.h> +#include <kdebug.h> + +#include "datablocks/mixednumber.h" +#include "datablocks/recipe.h" + +MXPImporter::MXPImporter() : BaseImporter() +{} + +void MXPImporter::parseFile( const TQString &file ) +{ + TQFile input( file ); + + if ( input.open( IO_ReadOnly ) ) { + TQTextStream stream( &input ); + stream.skipWhiteSpace(); + + TQString line; + while ( !stream.atEnd() ) { + line = stream.readLine().stripWhiteSpace(); + + if ( line.simplifyWhiteSpace().contains( "Exported from MasterCook" ) ) { + importMXP( stream ); + } + else if ( line == "{ Exported from MasterCook Mac }" ) { + importMac( stream ); + } + else if ( line == "@@@@@" ) { + importGeneric( stream ); + } + + stream.skipWhiteSpace(); + } + + if ( fileRecipeCount() == 0 ) + addWarningMsg( i18n( "No recipes found in this file." ) ); + } + else + setErrorMsg( i18n( "Unable to open file." ) ); +} + +MXPImporter::~MXPImporter() +{} + +void MXPImporter::importMXP( TQTextStream &stream ) +{ + Recipe recipe; + + kapp->processEvents(); //don't want the user to think its frozen... especially for files with thousands of recipes + + //kdDebug()<<"Found recipe MXP format: * Exported from MasterCook *"<<endl; + TQString current; + + // title + stream.skipWhiteSpace(); + recipe.title = stream.readLine().stripWhiteSpace(); + //kdDebug()<<"Found title: "<<m_title<<endl; + + //author + stream.skipWhiteSpace(); + current = stream.readLine().stripWhiteSpace(); + if ( current.mid( 0, current.find( ":" ) ).simplifyWhiteSpace().lower() == "recipe by" ) { + Element new_author( current.mid( current.find( ":" ) + 1, current.length() ).stripWhiteSpace() ); + recipe.authorList.append( new_author ); + //kdDebug()<<"Found author: "<<new_author.name<<endl; + } + else { + addWarningMsg( TQString( i18n( "While loading recipe \"%1\" " + "the field \"Recipe By:\" is either missing or could not be detected." ) ).arg( recipe.title ) ); + } + + //servings + stream.skipWhiteSpace(); + current = stream.readLine().stripWhiteSpace(); + if ( current.mid( 0, current.find( ":" ) ).simplifyWhiteSpace().lower() == "serving size" ) { + //allows serving size to be loaded even if preparation time is missing + int end_index; + if ( current.contains( "preparation time", FALSE ) ) + end_index = current.find( "preparation time", 0, FALSE ) - 15; + else + end_index = current.length(); + + recipe.yield.amount = current.mid( current.find( ":" ) + 1, end_index ).stripWhiteSpace().toInt(); + recipe.yield.type = i18n("servings"); + //kdDebug()<<"Found serving size: "<<recipe.yield.amount<<endl; + } + else { + addWarningMsg( TQString( i18n( "While loading recipe \"%1\" " + "the field \"Serving Size:\" is either missing or could not be detected." ) ).arg( recipe.title ) ); + } + + if ( current.contains( "preparation time", FALSE ) ) { + TQString prep_time = current.mid( current.find( ":", current.find( "preparation time", 0, FALSE ) ) + 1, + current.length() ).stripWhiteSpace(); + recipe.prepTime = TQTime( prep_time.section( ':', 0, 0 ).toInt(), prep_time.section( ':', 1, 1 ).toInt() ); + kdDebug() << "Found preparation time: " << prep_time << endl; + } + else { + addWarningMsg( TQString( i18n( "While loading recipe \"%1\" " + "the field \"Preparation Time:\" is either missing or could not be detected." ) ).arg( recipe.title ) ); + } + + loadCategories( stream, recipe ); + loadIngredients( stream, recipe ); + loadInstructions( stream, recipe ); + loadOptionalFields( stream, recipe ); + + add + ( recipe ); + + if ( !stream.atEnd() ) { + importMXP( stream ); + return ; + } +} + +void MXPImporter::loadCategories( TQTextStream &stream, Recipe &recipe ) +{ + //====================categories====================// + stream.skipWhiteSpace(); + TQString current = stream.readLine().stripWhiteSpace(); + if ( current.mid( 0, current.find( ":" ) ).simplifyWhiteSpace().lower() == "categories" ) { + TQString tmp_str = current.mid( current.find( ":" ) + 1, current.length() ).stripWhiteSpace(); + + while ( current.stripWhiteSpace() != "Amount Measure Ingredient -- Preparation Method" && !stream.atEnd() ) { + if ( !tmp_str.isEmpty() ) { + TQStringList categories = TQStringList::split( " ", tmp_str ); + for ( TQStringList::const_iterator it = categories.begin(); it != categories.end(); ++it ) { + Element new_cat( ( *it ).stripWhiteSpace() ); + recipe.categoryList.append( new_cat ); + + //kdDebug()<<"Found category: "<<new_cat.name<<endl; + } + } + + current = stream.readLine(); + tmp_str = current; + } + //else + // kdDebug()<<"No categories found."<<endl; + } + else { + addWarningMsg( TQString( i18n( "While loading recipe \"%1\" " + "the field \"Categories:\" is either missing or could not be detected." ) ).arg( recipe.title ) ); + + //the ingredient loaded will expect the last thing to have been read to be this header line + while ( current.stripWhiteSpace() != "Amount Measure Ingredient -- Preparation Method" && !stream.atEnd() ) + current = stream.readLine(); + } +} + +void MXPImporter::loadIngredients( TQTextStream &stream, Recipe &recipe ) +{ + //============ingredients=================// + stream.skipWhiteSpace(); + ( void ) stream.readLine(); + TQString current = stream.readLine(); + if ( !current.contains( "NONE" ) && !current.isEmpty() ) { + while ( !current.isEmpty() && !stream.atEnd() ) { + Ingredient new_ingredient; + + //amount + TQString amount_str = current.mid( 0, 9 ).simplifyWhiteSpace(); + if ( !amount_str.isEmpty() ) // case of amount_str.isEmpty() correctly handled by class default + { + bool ok; + MixedNumber amount( MixedNumber::fromString( amount_str, &ok ) ); + if ( !ok ) + { + addWarningMsg( TQString( i18n( "While loading recipe \"%1\" Invalid amount \"%2\" in the line \"%3\"" ) ).arg( recipe.title ).arg( amount_str ).arg( current.stripWhiteSpace() ) ); + current = stream.readLine(); + continue; + } + new_ingredient.amount = amount.toDouble(); + } + + //units + TQString units( current.mid( 9, 13 ) ); + new_ingredient.units = Unit( units.simplifyWhiteSpace(), new_ingredient.amount ); + + //name + int dash_index = current.find( "--" ); + + int length; + if ( dash_index == -1 || dash_index == 24 ) //ignore a dash in the first position (index 24) + length = current.length(); + else + length = dash_index - 22; + + TQString ingredient_name( current.mid( 22, length ) ); + new_ingredient.name = ingredient_name.stripWhiteSpace(); + + //prep method + if ( dash_index != -1 && dash_index != 24 ) //ignore a dash in the first position (index 24) + new_ingredient.prepMethodList.append( Element(current.mid( dash_index + 2, current.length() ).stripWhiteSpace()) ); + + recipe.ingList.append( new_ingredient ); + //kdDebug()<<"Found ingredient: amount="<<new_ingredient.amount + // <<", unit:"<<new_ingredient.units + // <<", name:"<<new_ingredient.name + // <<", prep_method:"<<prep_method<<endl; + + current = stream.readLine(); + } + } + //else + // kdDebug()<<"No ingredients found."<<endl; +} + +void MXPImporter::loadInstructions( TQTextStream &stream, Recipe &recipe ) +{ + //==========================instructions ( along with other optional fields... mxp format doesn't define end of ingredients and start of other fields )==============// + stream.skipWhiteSpace(); + TQString current = stream.readLine().stripWhiteSpace(); + while ( !current.contains( "- - - -" ) && !stream.atEnd() ) { + if ( current.stripWhiteSpace() == "Source:" ) { + Element new_author( getNextQuotedString( stream ) ); + recipe.authorList.append( new_author ); + //kdDebug()<<"Found source: "<<new_author.name<<" (adding as author)"<<endl; + } + else if ( current.stripWhiteSpace() == "Description:" ) { + TQString description = getNextQuotedString( stream ); + //kdDebug()<<"Found description: "<<m_description<<" (adding to end of instructions)"<<endl; + recipe.instructions += "\n\nDescription: " + description; + } + else if ( current.stripWhiteSpace() == "S(Internet Address):" ) { + TQString internet = getNextQuotedString( stream ); + //kdDebug()<<"Found internet address: "<<m_internet<<" (adding to end of instructions)"<<endl; + recipe.instructions += "\n\nInternet address: " + internet; + } + else if ( current.stripWhiteSpace() == "Yield:" ) { + recipe.yield.amount = getNextQuotedString( stream ).stripWhiteSpace().toInt(); + recipe.yield.type = i18n("servings"); + //kdDebug()<<"Found yield: "<<recipe.yield.amount<<" (adding as servings)"<<endl; + } + else if ( current.stripWhiteSpace() == "T(Cook Time):" ) { + ( void ) getNextQuotedString( stream ); //this would be prep time, but we don't use prep time at the moment + //kdDebug()<<"Found cook time: "<<m_prep_time<<" (adding as prep time)"<<endl; + } + else if ( current.stripWhiteSpace() == "Cuisine:" ) { + Element new_cat( getNextQuotedString( stream ) ); + recipe.categoryList.append( new_cat ); + //kdDebug()<<"Found cuisine (adding as category): "<<new_cat.name<<endl; + } + else + recipe.instructions += current + "\n"; + + current = stream.readLine().stripWhiteSpace(); + } + recipe.instructions = recipe.instructions.stripWhiteSpace(); + //kdDebug()<<"Found instructions: "<<m_instructions<<endl; +} + +void MXPImporter::loadOptionalFields( TQTextStream &stream, Recipe &recipe ) +{ + //=================after here, fields are optional=========================// + stream.skipWhiteSpace(); + TQString current = stream.readLine().stripWhiteSpace(); + + TQString notes; + + //Note: we simplifyWhiteSpace() because some versions of MasterCook have "Exported from MasterCook" and others have "Exported from MasterCook". + // This also could work around a typo or such. + while ( !current.simplifyWhiteSpace().contains( "Exported from MasterCook" ) && !stream.atEnd() ) { + //suggested wine + if ( current.mid( 0, current.find( ":" ) ).simplifyWhiteSpace().lower() == "suggested wine" ) { + TQString wine = current.mid( current.find( ":" ) + 1, current.length() ).stripWhiteSpace(); + //kdDebug()<<"Found suggested wine: "<<m_wine<<" (adding to end of instructions)"<<endl; + + recipe.instructions += "\n\nSuggested wine: " + wine; + } + //Nutr. Assoc. + if ( current.mid( 0, current.find( ":" ) ).simplifyWhiteSpace().lower() == "nutr. assoc." ) { + TQString nutr_assoc = current.mid( current.find( ":" ) + 1, current.length() ).stripWhiteSpace(); + //kdDebug()<<"Found nutrient association: "<<nutr_assoc<<" (adding to end of instructions)"<<endl; + + recipe.instructions += "\n\nNutrient Association: " + nutr_assoc; + } + else if ( current.mid( 0, current.find( ":" ) ).simplifyWhiteSpace().lower() == "per serving (excluding unknown items)" ) { //per serving... maybe we can do something with this info later + TQString per_serving_info = current.mid( current.find( ":" ) + 1, current.length() ).stripWhiteSpace(); + //kdDebug()<<"Found per serving (excluding unknown items): "<<per_serving_info<<" (adding to end of instructions)"<<endl; + + recipe.instructions += "\n\nPer Serving (excluding unknown items): " + per_serving_info; + } + else if ( current.mid( 0, current.find( ":" ) ).simplifyWhiteSpace().lower() == "per serving" ) { //per serving... maybe we can do something with this info later + TQString per_serving_info = current.mid( current.find( ":" ) + 1, current.length() ).stripWhiteSpace(); + //kdDebug()<<"Found per serving: "<<per_serving_info<<" (adding to end of instructions)"<<endl; + + recipe.instructions += "\n\nPer Serving: " + per_serving_info; + } + else if ( current.mid( 0, current.find( ":" ) ).simplifyWhiteSpace().lower() == "food exchanges" ) { //food exchanges... maybe we can do something with this info later + TQString food_exchange_info = current.mid( current.find( ":" ) + 1, current.length() ).stripWhiteSpace(); + //kdDebug()<<"Found food exchanges: "<<food_exchange_info<<" (adding to end of instructions)"<<endl; + + recipe.instructions += "\n\nFood Exchanges: " + food_exchange_info; + } + else if ( current.mid( 0, current.find( ":" ) ).simplifyWhiteSpace().lower() == "serving ideas" ) { //serving ideas + TQString serving_ideas = current.mid( current.find( ":" ) + 1, current.length() ).stripWhiteSpace(); + //kdDebug()<<"Found serving ideas: "<<m_serving_ideas<<" (adding to end of instructions)"<<endl; + + recipe.instructions += "\n\nServing ideas: " + serving_ideas; + } + else if ( current.mid( 0, current.find( ":" ) ).simplifyWhiteSpace().lower() == "notes" ) //notes + notes = current.mid( current.find( ":" ) + 1, current.length() ).stripWhiteSpace(); + else if ( !current.isEmpty() && current != "_____" ) //if it doesn't belong to any other field, assume it a part of a multi-line notes field + notes += "\n" + current; + + current = stream.readLine().stripWhiteSpace(); + } + + /*possible fields to implement later: + + Nutr. Assoc. : 0 0 0 0 0 + + Ratings : Cholesterol Rating 5 Complete Meal 3 + Cost 3 Depth 3 + Difficulty 2 Fanciness 7 + Fat Content 5 Good For Crowds 10 + Intensity 5 Intricacy 2 + Kid Appeal 3 Looks 5 + Portability 3 Richness 7 + Serving Temperature 8 Spicy Hotness 2 + Tartness 7 + + */ + if ( !notes.isNull() ) { + //kdDebug()<<TQString("Found notes: %s (adding to end of instructions)").arg(m_notes)<<endl; + recipe.instructions += "\n\nNotes: " + notes.stripWhiteSpace(); + } +} + +void MXPImporter::importGeneric( TQTextStream & /*stream*/ ) +{ + setErrorMsg( i18n( "MasterCook's Generic Export format is currently not supported. Please write to [email protected] to request support for this format." ) ); + //not even sure it this is worth writing... its rather obsolete +} + +void MXPImporter::importMac( TQTextStream & /*stream*/ ) +{ + setErrorMsg( i18n( "MasterCook Mac's Export format is currently not supported. Please write to [email protected] to request support for this format." ) ); + //not even sure it this is worth writing... its rather obsolete +} + +TQString MXPImporter::getNextQuotedString( TQTextStream &stream ) +{ + stream.skipWhiteSpace(); + TQString current = stream.readLine().stripWhiteSpace(); + TQString return_str; + + if ( current.left( 1 ) == "\"" ) + return_str = current.mid( 1, current.length() - 1 ); + else + return current; + + while ( current.right( 1 ) != "\"" && !stream.atEnd() ) { + current = stream.readLine().stripWhiteSpace(); + return_str += "\n" + current; + } + + //take off quote at end + return_str = return_str.mid( 0, return_str.length() - 1 ); + + return return_str.stripWhiteSpace(); +} diff --git a/src/importers/mxpimporter.h b/src/importers/mxpimporter.h new file mode 100644 index 0000000..6bdee22 --- /dev/null +++ b/src/importers/mxpimporter.h @@ -0,0 +1,45 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef MXPIMPORTER_H +#define MXPIMPORTER_H + +#include <tqstring.h> + +#include "baseimporter.h" + +/** Class to import MasterCook's MXP (MasterCook Export) file format. + * This is a human-readable format used in Mastercook up until version 4. + * @author Jason Kivlighn + */ +class MXPImporter : public BaseImporter +{ +public: + MXPImporter(); + virtual ~MXPImporter(); + +protected: + void parseFile( const TQString& filename ); + +private: + void importMXP( TQTextStream &stream ); + + void loadCategories( TQTextStream &stream, Recipe &recipe ); + void loadIngredients( TQTextStream &stream, Recipe &recipe ); + void loadInstructions( TQTextStream &stream, Recipe &recipe ); + void loadOptionalFields( TQTextStream &stream, Recipe &recipe ); + + void importMac( TQTextStream &stream ); + void importGeneric( TQTextStream &stream ); + + TQString getNextQuotedString( TQTextStream &stream ); +}; + +#endif //MXPIMPORTER_H diff --git a/src/importers/nycgenericimporter.cpp b/src/importers/nycgenericimporter.cpp new file mode 100644 index 0000000..30e5bdf --- /dev/null +++ b/src/importers/nycgenericimporter.cpp @@ -0,0 +1,196 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "nycgenericimporter.h" + +#include <tdeapplication.h> +#include <tdelocale.h> +#include <kdebug.h> + +#include <tqfile.h> +#include <tqtextstream.h> +#include <tqstringlist.h> +#include <tqregexp.h> + +#include "datablocks/mixednumber.h" +#include "datablocks/recipe.h" + +NYCGenericImporter::NYCGenericImporter() : BaseImporter() +{} + +void NYCGenericImporter::parseFile( const TQString &file ) +{ + first = true; + + m_recipe.empty(); + + TQFile input( file ); + if ( input.open( IO_ReadOnly ) ) { + TQTextStream stream( &input ); + stream.skipWhiteSpace(); + + if ( !stream.atEnd() && stream.readLine().startsWith( "@@@@@" ) ) + importNYCGeneric( stream ); + else { + setErrorMsg( i18n( "File does not appear to be a valid NYC export." ) ); + return ; + } + } + else + setErrorMsg( i18n( "Unable to open file." ) ); +} + +NYCGenericImporter::~NYCGenericImporter() +{} + +void NYCGenericImporter::importNYCGeneric( TQTextStream &stream ) +{ + kapp->processEvents(); //don't want the user to think its frozen... especially for files with thousands of recipes + + TQString current; + + stream.skipWhiteSpace(); + + //title + while ( !( current = stream.readLine() ).isEmpty() && !stream.atEnd() ) + m_recipe.title = current; + + //categories + while ( !( current = stream.readLine() ).isEmpty() && !stream.atEnd() ) { + if ( current[ 0 ].isNumber() ) { + loadIngredientLine( current ); + break; + } //oops, this is really an ingredient line (there was no category line) + + TQStringList categories = TQStringList::split( ',', current ); + + if ( categories.count() > 0 && categories[ 0 ].upper() == "none" ) //there are no categories + break; + + for ( TQStringList::const_iterator it = categories.begin(); it != categories.end(); ++it ) { + Element new_cat( TQString( *it ).stripWhiteSpace() ); + kdDebug() << "Found category: " << new_cat.name << endl; + m_recipe.categoryList.append( new_cat ); + } + } + + //ingredients + while ( !( current = stream.readLine() ).isEmpty() && !stream.atEnd() ) + loadIngredientLine( current ); + + //everything else is the instructions with optional "contributor", "prep time" and "yield" + bool found_next; + while ( !( found_next = ( current = stream.readLine() ).startsWith( "@@@@@" ) ) && !stream.atEnd() ) { + if ( current.startsWith( "Contributor:" ) ) { + Element new_author( current.mid( current.find( ':' ) + 1, current.length() ).stripWhiteSpace() ); + kdDebug() << "Found author: " << new_author.name << endl; + m_recipe.authorList.append( new_author ); + } + else if ( current.startsWith( "Preparation Time:" ) ) { + m_recipe.prepTime = TQTime::fromString( current.mid( current.find( ':' ), current.length() ) ); + } + else if ( current.startsWith( "Yield:" ) ) { + int colon_index = current.find( ':' ); + int amount_type_sep_index = current.find(" ",colon_index+1); + + m_recipe.yield.amount = current.mid( colon_index+2, amount_type_sep_index-colon_index ).toDouble(); + m_recipe.yield.type = current.mid( amount_type_sep_index+3, current.length() ); + } + else if ( current.startsWith( "NYC Nutrition Analysis (per serving or yield unit):" ) ) { + //m_recipe.instructions += current + "\n"; + } + else if ( current.startsWith( "NYC Nutrilink:" ) ) { + //m_recipe.instructions += current + "\n"; + } + else if ( !current.stripWhiteSpace().isEmpty() && !current.startsWith("** Exported from Now You're Cooking!") ) { + m_recipe.instructions += current + "\n"; + } + } + + m_recipe.instructions = m_recipe.instructions.stripWhiteSpace(); + putDataInRecipe(); + + if ( found_next ) + importNYCGeneric( stream ); +} + +void NYCGenericImporter::putDataInRecipe() +{ + //put it in the recipe list + add( m_recipe ); + + //reset for the next recipe + m_recipe.empty(); +} + +void NYCGenericImporter::loadIngredientLine( const TQString &line ) +{ + TQString current = line; + + if ( current.contains( "-----" ) ) { + current_header = current.stripWhiteSpace(); + kdDebug() << "Found ingredient header: " << current_header << endl; + return ; + } + + MixedNumber amount( 0, 0, 1 ); + TQString unit; + TQString name; + TQString prep; + + TQStringList ingredient_line = TQStringList::split( ' ', current ); + + bool found_amount = false; + + if ( !ingredient_line.empty() ) //probably an unnecessary check... but to be safe + { + bool ok; + MixedNumber test_amount = MixedNumber::fromString( ingredient_line[ 0 ], &ok ); + if ( ok ) + { + amount = amount + test_amount; + ingredient_line.pop_front(); + found_amount = true; + } + } + if ( !ingredient_line.empty() ) //probably an unnecessary check... but to be safe + { + bool ok; + MixedNumber test_amount = MixedNumber::fromString( ingredient_line[ 0 ], &ok ); + if ( ok ) + { + amount = amount + test_amount; + ingredient_line.pop_front(); + found_amount = true; + } + } + + if ( found_amount ) { + unit = ingredient_line[ 0 ]; + ingredient_line.pop_front(); + } + + //now join each separate part of ingredient (name, unit, amount) + name = ingredient_line.join( " " ); + + int prep_sep_index = name.find( TQRegExp( "(--|,;;)" ) ); + if ( prep_sep_index == -1 ) + prep_sep_index = name.length(); + + name = name.left( prep_sep_index ).stripWhiteSpace(); + prep = name.mid( prep_sep_index+1, name.length() ).stripWhiteSpace(); + + Ingredient new_ingredient( name, amount.toDouble(), Unit( unit, amount.toDouble() ) ); + new_ingredient.group = current_header; + new_ingredient.prepMethodList = ElementList::split(",",prep); + m_recipe.ingList.append( new_ingredient ); + +} + diff --git a/src/importers/nycgenericimporter.h b/src/importers/nycgenericimporter.h new file mode 100644 index 0000000..6b3bc2f --- /dev/null +++ b/src/importers/nycgenericimporter.h @@ -0,0 +1,44 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef NYCGENERICIMPORTER_H +#define NYCGENERICIMPORTER_H + +#include <tqstring.h> +#include <tqdatetime.h> + +#include "baseimporter.h" +#include "datablocks/ingredientlist.h" +#include "datablocks/elementlist.h" + +/** Class to import The NYC (Now You're Cooking) Generic Export file format. + * @author Jason Kivlighn + */ +class NYCGenericImporter : public BaseImporter +{ +public: + NYCGenericImporter(); + ~NYCGenericImporter(); + +protected: + void parseFile( const TQString& filename ); + +private: + void importNYCGeneric( TQTextStream &stream ); + void putDataInRecipe(); + void loadIngredientLine( const TQString & ); + + Recipe m_recipe; + TQString current_header; + + bool first; +}; + +#endif //NYCGENERICIMPORTER_H diff --git a/src/importers/recipemlimporter.cpp b/src/importers/recipemlimporter.cpp new file mode 100644 index 0000000..8866648 --- /dev/null +++ b/src/importers/recipemlimporter.cpp @@ -0,0 +1,376 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Richard L�rk�ng * +* * +* Copyright (C) 2003-2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "recipemlimporter.h" + +#include <tqfile.h> +#include <tqdatetime.h> + +#include <tdelocale.h> +#include <kdebug.h> + +#include "datablocks/recipe.h" +#include "datablocks/mixednumber.h" + +RecipeMLImporter::RecipeMLImporter() : BaseImporter() +{} + +void RecipeMLImporter::parseFile( const TQString& file ) +{ + TQFile input( file ); + if ( input.open( IO_ReadOnly ) ) { + TQDomDocument doc; + TQString error; + int line; + int column; + if ( !doc.setContent( &input, &error, &line, &column ) ) { + setErrorMsg( TQString( i18n( "\"%1\" at line %2, column %3. This may not be a RecipeML file." ) ).arg( error ).arg( line ).arg( column ) ); + return ; + } + + TQDomElement recipeml = doc.documentElement(); + + if ( recipeml.tagName() != "recipeml" ) { + setErrorMsg( i18n( "This file does not appear to be a valid RecipeML archive." ) ); + return ; + } + + TQDomNodeList l = recipeml.childNodes(); + + for ( unsigned i = 0 ; i < l.count(); i++ ) { + TQDomElement el = l.item( i ).toElement(); + TQString tagName = el.tagName(); + + if ( tagName == "meta" ) + continue; + else if ( tagName == "recipe" ) + readRecipemlRecipe( el ); + else if ( tagName == "menu" ) + readRecipemlMenu( el ); + else + kdDebug() << "Unknown tag within <recipeml>: " << tagName << endl; + } + } + else + setErrorMsg( i18n( "Unable to open file." ) ); +} + +RecipeMLImporter::~RecipeMLImporter() +{} + +void RecipeMLImporter::readRecipemlRecipe( const TQDomElement& recipe_element ) +{ + recipe.empty(); + + TQDomNodeList l = recipe_element.childNodes(); + + for ( unsigned i = 0; i < l.count(); i++ ) { + TQDomElement el = l.item( i ).toElement(); + TQString tagName = el.tagName(); + + if ( tagName == "head" ) + readRecipemlHead( el ); + else if ( tagName == "ingredients" ) + readRecipemlIngs( el ); + else if ( tagName == "directions" ) + readRecipemlDirections( el ); + else if ( tagName == "description" ) {} //TODO: what do we do with this? + else if ( tagName == "equipment" ) {} //TODO: what do we do with this? + else if ( tagName == "nutrition" ) {} //TODO: what do we do with this? + else if ( tagName == "diet-exchanges" ) {} //TODO: what do we do with this? + else + kdDebug() << "Unknown tag within <recipe>: " << el.tagName() << endl; + } + + add + ( recipe ); +} + +void RecipeMLImporter::readRecipemlHead( const TQDomElement& head ) +{ + TQDomNodeList l = head.childNodes(); + for ( unsigned i = 0 ; i < l.count(); i++ ) { + TQDomElement el = l.item( i ).toElement(); + TQString tagName = el.tagName(); + + if ( tagName == "title" ) + recipe.title = el.text().stripWhiteSpace(); + else if ( tagName == "subtitle" ) + recipe.title += ": " + el.text().stripWhiteSpace(); + else if ( tagName == "version" ) {} //TODO: what do we do with this? + else if ( tagName == "source" ) + readRecipemlSrcItems( el ); + else if ( tagName == "categories" ) { + TQDomNodeList categories = el.childNodes(); + for ( unsigned j = 0; j < categories.count(); j++ ) { + TQDomElement c = categories.item( j ).toElement(); + if ( c.tagName() == "cat" ) { + recipe.categoryList.append( Element( c.text() ) ); + } + } + } + else if ( tagName == "description" ) + recipe.instructions += "\n\nDescription: " + el.text().stripWhiteSpace(); + else if ( tagName == "preptime" ) + readRecipemlPreptime( el ); + else if ( tagName == "yield" ) { + TQDomNodeList yieldChildren = el.childNodes(); + for ( unsigned j = 0; j < yieldChildren.count(); j++ ) { + TQDomElement y = yieldChildren.item( j ).toElement(); + TQString tagName = y.tagName(); + if ( tagName == "range" ) + readRecipemlRange( y, recipe.yield.amount, recipe.yield.amount_offset ); + else if ( tagName == "unit" ) + recipe.yield.type = y.text(); + else + kdDebug() << "Unknown tag within <yield>: " << y.tagName() << endl; + } + } + else + kdDebug() << "Unknown tag within <head>: " << el.tagName() << endl; + } +} + +void RecipeMLImporter::readRecipemlIngs( const TQDomElement& ings ) +{ + TQDomNodeList l = ings.childNodes(); + for ( unsigned i = 0 ; i < l.count(); i++ ) { + TQDomElement el = l.item( i ).toElement(); + TQString tagName = el.tagName(); + + if ( tagName == "ing" ) + readRecipemlIng( el ); + else if ( tagName == "ing-div" ) //NOTE: this can have the "type" attribute + { + TQString header; + TQDomNodeList ingDiv = el.childNodes(); + for ( unsigned j = 0; j < ingDiv.count(); j++ ) + { + TQDomElement cEl = ingDiv.item( j ).toElement(); + if ( cEl.tagName() == "title" ) + header = cEl.text().stripWhiteSpace(); + else if ( cEl.tagName() == "description" ) {} //TODO: what do we do with this? + else if ( cEl.tagName() == "ing" ) + readRecipemlIng( cEl, 0, header ); + else if ( tagName == "note" ) {} //TODO: what do we do with this? + else + kdDebug() << "Unknown tag within <ing-div>: " << cEl.tagName() << endl; + } + } + else if ( tagName == "note" ) {} //TODO: what do we do with this? + else + kdDebug() << "Unknown tag within <ingredients>: " << el.tagName() << endl; + } +} + +void RecipeMLImporter::readRecipemlIng( const TQDomElement& ing, Ingredient *ing_parent, const TQString &header ) +{ + Ingredient new_ing; + + TQDomNodeList ingChilds = ing.childNodes(); + + TQString name, unit, size, prep_method; + Ingredient quantity; + quantity.amount = 1;// default quantity assumed by RecipeML DTD + + for ( unsigned j = 0; j < ingChilds.count(); j++ ) { + TQDomElement ingChild = ingChilds.item( j ).toElement(); + TQString tagName = ingChild.tagName(); + + if ( tagName == "amt" ) { + TQDomNodeList amtChilds = ingChild.childNodes(); + + for ( unsigned k = 0; k < amtChilds.count(); k++ ) { + TQDomElement amtChild = amtChilds.item( k ).toElement(); + + if ( amtChild.tagName() == "qty" ) + readRecipemlTQty( amtChild, quantity ); + else if ( amtChild.tagName() == "size" ) + size = amtChild.text().stripWhiteSpace(); + else if ( amtChild.tagName() == "unit" ) + unit = amtChild.text().stripWhiteSpace(); + else + kdDebug() << "Unknown tag within <amt>: " << amtChild.tagName() << endl; + } + } + else if ( tagName == "item" ) { + name = ingChild.text().stripWhiteSpace(); + if ( ing.attribute( "optional", "no" ) == "yes" ) + prep_method = "(optional)"; + } + else if ( tagName == "prep" ) { //FIXME: this overwrite the optional attribute + prep_method = ingChild.text().stripWhiteSpace(); + } + else if ( tagName == "alt-ing" ) + readRecipemlIng( ingChild, &new_ing, header ); + else + kdDebug() << "Unknown tag within <ing>: " << ingChild.tagName() << endl; + } + + if ( !size.isEmpty() ) + unit.prepend( size + " " ); + + new_ing.name = name; + new_ing.units = Unit( unit, quantity.amount+quantity.amount_offset ); + new_ing.amount = quantity.amount; + new_ing.amount_offset = quantity.amount_offset; + new_ing.group = header; + new_ing.prepMethodList = ElementList::split(",",prep_method); + + if ( !ing_parent ) + recipe.ingList.append(new_ing); + else + ing_parent->substitutes.append( new_ing ); +} + +void RecipeMLImporter::readRecipemlDirections( const TQDomElement& dirs ) +{ + TQDomNodeList l = dirs.childNodes(); + + TQStringList directions; + + for ( unsigned i = 0 ; i < l.count(); i++ ) { + TQDomElement el = l.item( i ).toElement(); + + if ( el.tagName() == "step" ) + directions.append( el.text().stripWhiteSpace() ); + else + kdDebug() << "Unknown tag within <directions>: " << el.tagName() << endl; + } + + TQString directionsText; + + if ( directions.count() > 1 ) { + for ( unsigned i = 1; i <= directions.count(); i++ ) { + if ( i != 1 ) { + directionsText += "\n\n"; + } + + TQString sWith = TQString( "%1. " ).arg( i ); + TQString text = directions[ i - 1 ]; + if ( !text.stripWhiteSpace().startsWith( sWith ) ) + directionsText += sWith; + directionsText += text; + } + } + else + directionsText = directions[ 0 ]; + + recipe.instructions = directionsText; +} + +void RecipeMLImporter::readRecipemlMenu( const TQDomElement& menu_el ) +{ + TQDomNodeList l = menu_el.childNodes(); + for ( unsigned i = 0 ; i < l.count(); i++ ) { + TQDomElement el = l.item( i ).toElement(); + TQString tagName = el.tagName(); + + if ( tagName == "head" ) + readRecipemlHead( el ); + else if ( tagName == "description" ) {} //TODO: what do we do with this? + else if ( tagName == "recipe" ) + readRecipemlRecipe( el ); + else + kdDebug() << "Unknown tag within <menu>: " << tagName << endl; + } +} + +void RecipeMLImporter::readRecipemlSrcItems( const TQDomElement& sources ) +{ + TQDomNodeList l = sources.childNodes(); + for ( unsigned i = 0 ; i < l.count(); i++ ) { + TQDomElement srcitem = l.item( i ).toElement(); + TQString tagName = srcitem.tagName(); + + if ( tagName == "srcitem" ) + recipe.authorList.append( Element( srcitem.text().stripWhiteSpace() ) ); + else + kdDebug() << "Unknown tag within <source>: " << tagName << endl; + } +} + +void RecipeMLImporter::readRecipemlPreptime( const TQDomElement &preptime ) +{ + TQDomNodeList l = preptime.childNodes(); + for ( unsigned i = 0 ; i < l.count(); i++ ) { + TQDomElement el = l.item( i ).toElement(); + TQString tagName = el.tagName(); + + if ( tagName == "time" ) { + int qty = 0; + TQString timeunit; + + TQDomNodeList time_l = el.childNodes(); + for ( unsigned i = 0 ; i < time_l.count(); i++ ) { + TQDomElement time_el = time_l.item( i ).toElement(); + TQString time_tagName = time_el.tagName(); + + if ( time_tagName == "qty" ) + qty = time_el.text().toInt(); + else if ( time_tagName == "timeunit" ) + timeunit = time_el.text(); + else + kdDebug() << "Unknown tag within <time>: " << time_tagName << endl; + } + + int minutes = 0; + int hours = 0; + if ( timeunit == "minutes" ) + minutes = qty; + else if ( timeunit == "hours" ) + hours = qty; + else + kdDebug() << "Unknown timeunit: " << timeunit << endl; + + recipe.prepTime = TQTime( hours + minutes / 60, minutes % 60 ); + } + else + kdDebug() << "Unknown tag within <preptime>: " << tagName << endl; + } +} + +void RecipeMLImporter::readRecipemlTQty( const TQDomElement &qty, Ingredient &ing ) +{ + TQDomNodeList qtyChilds = qty.childNodes(); + + for ( unsigned i = 0; i < qtyChilds.count(); i++ ) { + TQDomElement qtyChild = qtyChilds.item( i ).toElement(); + TQString tagName = qtyChild.tagName(); + if ( tagName == "range" ) + readRecipemlRange( qtyChild, ing.amount, ing.amount_offset ); + else if ( tagName.isEmpty() ) + ing.amount = MixedNumber::fromString( qty.text() ).toDouble(); + else + kdDebug() << "Unknown tag within <qty>: " << tagName << endl; + } +} + +void RecipeMLImporter::readRecipemlRange( const TQDomElement& range, double &amount, double &amount_offset ) +{ + TQDomNodeList rangeChilds = range.childNodes(); + double q1 = 1, q2 = 0; // default quantity assumed by RecipeML DTD + for ( unsigned j = 0; j < rangeChilds.count(); j++ ) { + TQDomElement rangeChild = rangeChilds.item( j ).toElement(); + TQString subTagName = rangeChild.tagName(); + if ( subTagName == "q1" ) + q1 = MixedNumber::fromString( rangeChild.text() ).toDouble(); + else if ( subTagName == "q2" ) + q2 = MixedNumber::fromString( rangeChild.text() ).toDouble(); + else + kdDebug() << "Unknown tag within <range>: " << subTagName << endl; + } + + amount = q1; + amount_offset = q2-q1; +} diff --git a/src/importers/recipemlimporter.h b/src/importers/recipemlimporter.h new file mode 100644 index 0000000..51c6f42 --- /dev/null +++ b/src/importers/recipemlimporter.h @@ -0,0 +1,53 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Richard L�rk�ng * +* * +* Copyright (C) 2003-2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + + +#ifndef RECIPEMLIMPORTER_H +#define RECIPEMLIMPORTER_H + +#include "baseimporter.h" +#include "datablocks/ingredient.h" +#include "datablocks/recipe.h" + +#include <tqdom.h> + +/** Class to import the RecipeML, XML-based file format. + * More info at http://www.formatdata.com/recipeml + * + * @author Jason Kivlighn + */ +class RecipeMLImporter : public BaseImporter +{ +public: + RecipeMLImporter(); + virtual ~RecipeMLImporter(); + +protected: + void parseFile( const TQString& filename ); + +private: + void readRecipemlDirections( const TQDomElement& dirs ); + void readRecipemlHead( const TQDomElement& head ); + void readRecipemlIng( const TQDomElement& ing1, Ingredient *ing2 = 0, const TQString &header = TQString::null ); + void readRecipemlIngs( const TQDomElement& ings ); + void readRecipemlMenu( const TQDomElement& menu ); + void readRecipemlSrcItems( const TQDomElement& sources ); + void readRecipemlRecipe( const TQDomElement& recipe ); + void readRecipemlPreptime( const TQDomElement &preptime ); + void readRecipemlTQty( const TQDomElement &qty, Ingredient &ing ); + void readRecipemlRange( const TQDomElement& range1, double &range2, double &range_offset ); + + Recipe recipe; +}; + +#endif //RECIPEMLIMPORTER_H diff --git a/src/importers/rezkonvimporter.cpp b/src/importers/rezkonvimporter.cpp new file mode 100644 index 0000000..58e59c0 --- /dev/null +++ b/src/importers/rezkonvimporter.cpp @@ -0,0 +1,291 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "rezkonvimporter.h" + +#include <tdeapplication.h> +#include <tdelocale.h> +#include <kdebug.h> + +#include <tqfile.h> +#include <tqregexp.h> +#include <tqtextstream.h> + +#include "datablocks/mixednumber.h" + +RezkonvImporter::RezkonvImporter() : BaseImporter() +{} + +RezkonvImporter::~RezkonvImporter() +{} + +void RezkonvImporter::parseFile( const TQString &filename ) +{ + TQFile input( filename ); + + if ( input.open( IO_ReadOnly ) ) { + TQTextStream stream( &input ); + stream.skipWhiteSpace(); + + TQString line; + + while ( !stream.atEnd() ) { + line = stream.readLine(); + + if ( line.contains( TQRegExp( "^=====.*REZKONV.*" ) ) ) { + TQStringList raw_recipe; + while ( !( line = stream.readLine() ).contains( TQRegExp( "^=====\\s*$" ) ) && !stream.atEnd() ) + raw_recipe << line; + + readRecipe( raw_recipe ); + } + } + + if ( fileRecipeCount() == 0 ) + setErrorMsg( i18n( "No recipes found in this file." ) ); + } + else + setErrorMsg( i18n( "Unable to open file." ) ); +} + +void RezkonvImporter::readRecipe( const TQStringList &raw_recipe ) +{ + kapp->processEvents(); //don't want the user to think its frozen... especially for files with thousands of recipes + + Recipe recipe; + + TQStringList::const_iterator text_it = raw_recipe.begin(); + m_end_it = raw_recipe.end(); + + //title (Titel) + text_it++; + recipe.title = ( *text_it ).mid( ( *text_it ).find( ":" ) + 1, ( *text_it ).length() ).stripWhiteSpace(); + kdDebug() << "Found title: " << recipe.title << endl; + + //categories (Kategorien): + text_it++; + TQStringList categories = TQStringList::split( ',', ( *text_it ).mid( ( *text_it ).find( ":" ) + 1, ( *text_it ).length() ) ); + for ( TQStringList::const_iterator it = categories.begin(); it != categories.end(); ++it ) { + Element new_cat; + new_cat.name = TQString( *it ).stripWhiteSpace(); + kdDebug() << "Found category: " << new_cat.name << endl; + recipe.categoryList.append( new_cat ); + } + + //yield (Menge) + text_it++; + //get the number between the ":" and the next space after it + TQString yield_str = ( *text_it ).stripWhiteSpace(); + yield_str.remove( TQRegExp( "^Menge:\\s*" ) ); + int sep_index = yield_str.find( ' ' ); + if ( sep_index != -1 ) + recipe.yield.type = yield_str.mid( sep_index+1 ); + readRange( yield_str.mid( 0, sep_index ), recipe.yield.amount, recipe.yield.amount_offset ); + kdDebug() << "Found yield: " << recipe.yield.amount << endl; + + bool is_sub = false; + bool last_line_empty = false; + text_it++; + while ( text_it != raw_recipe.end() ) { + if ( ( *text_it ).isEmpty() ) { + last_line_empty = true; + text_it++; + continue; + } + + if ( ( *text_it ).contains( TQRegExp( "^=====.*=$" ) ) ) //is a header + { + if ( ( *text_it ).contains( "quelle", false ) ) + { + loadReferences( text_it, recipe ); + break; //reference lines are the last before the instructions + } + else + loadIngredientHeader( *text_it, recipe ); + } + + //if it has no more than two spaces followed by a non-digit + //then we'll assume it is a direction line + else if ( last_line_empty && ( *text_it ).contains( TQRegExp( "^\\s{0,2}[^\\d\\s=]" ) ) ) + break; + else + loadIngredient( *text_it, recipe, is_sub ); + + last_line_empty = false; + text_it++; + } + + loadInstructions( text_it, recipe ); + + add + ( recipe ); + + current_header = TQString::null; +} + +void RezkonvImporter::loadIngredient( const TQString &string, Recipe &recipe, bool &is_sub ) +{ + Ingredient new_ingredient; + new_ingredient.amount = 0; //amount not required, so give default of 0 + + TQRegExp cont_test( "^-{1,2}" ); + if ( string.stripWhiteSpace().contains( cont_test ) ) { + TQString name = string.stripWhiteSpace(); + name.remove( cont_test ); + kdDebug() << "Appending to last ingredient: " << name << endl; + if ( !recipe.ingList.isEmpty() ) //so it doesn't crash when the first ingredient appears to be a continuation of another + ( *recipe.ingList.at( recipe.ingList.count() - 1 ) ).name += " " + name; + + return ; + } + + //amount + if ( !string.mid( 0, 7 ).stripWhiteSpace().isEmpty() ) + readRange( string.mid( 0, 7 ), new_ingredient.amount, new_ingredient.amount_offset ); + + //unit + TQString unit_str = string.mid( 8, 9 ).stripWhiteSpace(); + new_ingredient.units = Unit( unit_str, new_ingredient.amount ); + + //name and preparation method + new_ingredient.name = string.mid( 18, string.length() - 18 ).stripWhiteSpace(); + + //separate out the preparation method + TQString name_and_prep = new_ingredient.name; + int separator_index = name_and_prep.find( "," ); + if ( separator_index != -1 ) { + new_ingredient.name = name_and_prep.mid( 0, separator_index ).stripWhiteSpace(); + new_ingredient.prepMethodList = ElementList::split(",",name_and_prep.mid( separator_index + 1, name_and_prep.length() ).stripWhiteSpace() ); + } + + //header (if present) + new_ingredient.group = current_header; + + bool last_is_sub = is_sub; + if ( new_ingredient.prepMethodList.last().name == "or" ) { + new_ingredient.prepMethodList.pop_back(); + is_sub = true; + } + else + is_sub = false; + + if ( last_is_sub ) + ( *recipe.ingList.at( recipe.ingList.count() - 1 ) ).substitutes.append(new_ingredient); + else + recipe.ingList.append( new_ingredient ); +} + +void RezkonvImporter::loadIngredientHeader( const TQString &string, Recipe &/*recipe*/ ) +{ + TQString header = string; + header.remove( TQRegExp( "^=*" ) ).remove( TQRegExp( "=*$" ) ); + header = header.stripWhiteSpace(); + + kdDebug() << "found ingredient header: " << header << endl; + + current_header = header; +} + +void RezkonvImporter::loadInstructions( TQStringList::const_iterator &text_it, Recipe &recipe ) +{ + TQString instr; + TQRegExp rx_title( "^:{0,1}\\s*O-Titel\\s*:" ); + TQString line; + while ( text_it != m_end_it ) { + line = *text_it; + + //titles longer than the line width are rewritten here + if ( line.contains( rx_title ) ) { + line.remove( rx_title ); + recipe.title = line.stripWhiteSpace(); + + TQRegExp rx_line_cont( ":\\s*>{0,1}\\s*:" ); + while ( ( line = *text_it ).contains( rx_line_cont ) ) { + line.remove( rx_line_cont ); + recipe.title += line; + + text_it++; + } + kdDebug() << "Found long title: " << recipe.title << endl; + } + else { + if ( line.isEmpty() ) + instr += "\n\n"; + + instr += line; + } + + text_it++; + } + + recipe.instructions = instr; +} + +void RezkonvImporter::loadReferences( TQStringList::const_iterator &text_it, Recipe &recipe ) +{ + kdDebug() << "Found source header" << endl; + + while ( text_it != m_end_it ) { + text_it++; + TQRegExp rx_line_begin( "^\\s*-{0,2}\\s*" ); + + TQRegExp rx_creation_date = TQRegExp( "^\\s*-{0,2}\\s*Erfasst \\*RK\\*", false ); + if ( ( *text_it ).contains( rx_creation_date ) ) // date followed by typist + { + TQString date = *text_it; + date.remove( rx_creation_date ).remove( TQRegExp( " von\\s*$" ) ); + + // Date is given as DD.MM.YY + TQString s = date.section( ".", 0, 0 ); + int day = s.toInt(); + + s = date.section( ".", 1, 1 ); + int month = s.toInt(); + + s = date.section( ".", 2, 2 ); + int year = s.toInt(); + year += 1900; + if ( year < 1970 ) + year += 100; //we'll assume nothing has been created before 1970 (y2k issues :p) + + recipe.ctime = TQDate(year,month,day); + + #if 0 + //typist + text_it++; + TQString typist = = *text_it; + typist.remove( rx_line_begin ); + #endif + + } + else //everything else is an author + { + if ( ( *text_it ).contains( rx_line_begin ) ) { + TQString author = *text_it; + author.remove( rx_line_begin ); + + recipe.authorList.append( Element( author ) ); + } + else + break; + } + } +} + +void RezkonvImporter::readRange( const TQString &range_str, double &amount, double &amount_offset ) +{ + TQString from = range_str.section( '-', 0, 0 ); + TQString to = range_str.section( '-', 1, 1 ); + + amount = MixedNumber::fromString( from ).toDouble(); + + if ( !to.stripWhiteSpace().isEmpty() ) + amount_offset = MixedNumber::fromString( to ).toDouble() - amount; +} diff --git a/src/importers/rezkonvimporter.h b/src/importers/rezkonvimporter.h new file mode 100644 index 0000000..7f22255 --- /dev/null +++ b/src/importers/rezkonvimporter.h @@ -0,0 +1,42 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef REZKONVIMPORTER_H +#define REZKONVIMPORTER_H + +#include <tqstringlist.h> + +#include "baseimporter.h" +#include "datablocks/recipe.h" + +class RezkonvImporter : public BaseImporter +{ +public: + RezkonvImporter(); + ~RezkonvImporter(); + +protected: + void parseFile( const TQString &filename ); + +private: + void loadIngredient( const TQString &line, Recipe & recipe, bool &is_sub ); + void loadIngredientHeader( const TQString &line, Recipe & recipe ); + void loadInstructions( TQStringList::const_iterator &raw_text, Recipe & recipe ); + void loadReferences( TQStringList::const_iterator &raw_text, Recipe & recipe ); + + void readRange( const TQString &, double &amount, double &amount_offset ); + void readRecipe( const TQStringList &raw_text ); + + TQStringList::const_iterator m_end_it; + + TQString current_header; +}; + +#endif //REZKONVIMPORTER_H diff --git a/src/klomanager.cpp b/src/klomanager.cpp new file mode 100644 index 0000000..ac2639c --- /dev/null +++ b/src/klomanager.cpp @@ -0,0 +1,218 @@ +/*************************************************************************** +* Copyright (C) 2006 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "klomanager.h" + +#include <kdebug.h> + +#include <tqdom.h> +#include <tqfile.h> +#include <tqstringlist.h> + +KLOManager::KLOManager() +{ +} + +KLOManager::~KLOManager() +{ + +} + +TQStringList KLOManager::classes() +{ + TQStringList classesList; + classesList << "title" << "instructions" << "yield" << "prep_time" << "photo" << "authors" << + "categories" << "header" << "ingredients" << "properties" << "ratings"; + return classesList; +} + +void KLOManager::processDocument( const TQDomDocument &doc ) +{ + TQDomElement layout = doc.documentElement(); + + if ( layout.tagName() != "krecipes-layout" ) { + kdDebug() << "This file does not appear to be a valid Krecipes layout file." << endl; + return ; + } + + TQDomNodeList l = layout.childNodes(); + for ( unsigned int i = 0 ; i < l.count(); i++ ) { + TQDomElement el = l.item( i ).toElement(); + TQString tagName = el.tagName(); + TQDomNodeList subList = el.childNodes(); + /*if ( !*/beginObject( tagName )/* ) {*/; //###: just a thought.... + for ( unsigned int j = 0 ; j < subList.count(); j++ ) { + TQDomElement subEl = subList.item( j ).toElement(); + TQString subTagName = subEl.tagName(); + + if ( subTagName == "background-color" ) + loadBackgroundColor( tagName, getColorAttribute(el,subTagName) ); + else if ( subTagName == "font" ) + loadFont( tagName, getFontAttribute(el,subTagName) ); + else if ( subTagName == "text-color" ) + loadTextColor( tagName, getColorAttribute(el,subTagName) ); + else if ( subTagName == "visible" ) + loadVisibility( tagName, getBoolAttribute(el,subTagName) ); + else if ( subTagName == "alignment" ) + loadAlignment( tagName, getIntAttribute(el,subTagName) ); + else if ( subTagName == "border" ) + loadBorder( tagName, getBorderAttribute(el,subTagName) ); + else if ( subTagName == "columns" ) + loadColumns( tagName, getIntAttribute(el,subTagName) ); + else + kdDebug() << "Warning: Unknown tag within <" << tagName << ">: " << subTagName << endl; + } + endObject(); + } +} + +TQDomElement KLOManager::getLayoutAttribute( const TQDomElement &object, const TQString &attribute ) const +{ + TQDomNodeList l = object.childNodes(); + for ( unsigned i = 0; i < l.count(); i++ ) { + TQDomElement el = l.item( i ).toElement(); + + if ( el.tagName() == attribute ) + return el; + } + + kdDebug() << "Warning: Requested attribute \"" << attribute << "\" not found." << endl; + return TQDomElement(); +} + +bool KLOManager::getBoolAttribute( const TQDomElement &object, const TQString &attribute, bool defaultValue ) const +{ + TQDomElement result = getLayoutAttribute( object, attribute ); + if ( result.isNull() ) { + return defaultValue; + } + else { + return result.text() == "true"; + } +} + +TQColor KLOManager::getColorAttribute( const TQDomElement &object, const TQString &attribute, const TQColor &defaultValue ) const +{ + TQDomElement result = getLayoutAttribute( object, attribute ); + if ( result.isNull() ) { + return defaultValue; + } + else { + return TQColor(result.text()); + } +} + +TQString KLOManager::getTextAttribute( const TQDomElement &object, const TQString &attribute, const TQString &defaultValue ) const +{ + TQDomElement result = getLayoutAttribute( object, attribute ); + if ( result.isNull() ) { + return defaultValue; + } + else { + return result.text(); + } +} + +int KLOManager::getIntAttribute( const TQDomElement &object, const TQString &attribute, int defaultValue ) const +{ + TQDomElement result = getLayoutAttribute( object, attribute ); + if ( result.isNull() ) { + return defaultValue; + } + else { + return result.text().toInt(); + } +} + +KreBorder KLOManager::getBorderAttribute( const TQDomElement &object, const TQString &attribute, const KreBorder &defaultValue ) const +{ + TQDomElement result = getLayoutAttribute( object, attribute ); + if ( result.isNull() ) { + return defaultValue; + } + else { + return KreBorder( result.attribute( "width" ).toInt(), result.attribute( "style" ), TQColor(result.attribute( "color" )) ); + } +} + +TQFont KLOManager::getFontAttribute( const TQDomElement &object, const TQString &attribute, const TQFont &defaultValue ) const +{ + TQDomElement result = getLayoutAttribute( object, attribute ); + if ( result.isNull() ) { + return defaultValue; + } + else { + TQFont font; + font.fromString(result.text()); + return font; + } +} + +TQString KLOManager::alignmentAsCSS( int alignment ) +{ + TQString text; + + if ( alignment & TQt::AlignLeft ) + text += "text-align: left;\n"; + if ( alignment & TQt::AlignRight ) + text += "text-align: right;\n"; + if ( alignment & TQt::AlignHCenter ) + text += "text-align: center;\n"; + if ( alignment & TQt::AlignTop ) + text += "vertical-align: top;\n"; + if ( alignment & TQt::AlignBottom ) + text += "vertical-align: bottom;\n"; + if ( alignment & TQt::AlignVCenter ) + text += "vertical-align: middle;\n"; + + return text; +} + +TQString KLOManager::borderAsCSS( const KreBorder &border ) +{ + return TQString( "border: %1px %2 %3;\n" ).arg(border.width).arg(border.style).arg(border.color.name()); +} + +TQString KLOManager::bgColorAsCSS( const TQColor &color ) +{ + return TQString( "background-color: %1;\n" ).arg( color.name() ); +} + +TQString KLOManager::fontAsCSS( const TQFont &font ) +{ + TQString text; + + text += TQString( "font-family: %1;\n" ).arg( font.family() ); + text += TQString( "font-weight: %1;\n" ).arg( font.weight() ); + text += TQString( "font-size: %1pt;\n" ).arg( font.pointSize() ); + if ( font.underline() ) + text += "text-decoration: underline;\n"; + if ( font.strikeOut() ) + text += "text-decoration: line-through;\n"; + if ( font.bold() ) + text += "font-weight: bold;\n"; + if ( font.italic() ) + text += "font-style: italic;\n"; + + return text; +} + +TQString KLOManager::textColorAsCSS( const TQColor &color ) +{ + return TQString( "color: %1;\n" ).arg( color.name() ); +} + +TQString KLOManager::visibilityAsCSS( bool visible ) +{ + if ( visible ) + return "visibility: visible;\n"; + else + return "visibility: hidden;\n"; +} diff --git a/src/klomanager.h b/src/klomanager.h new file mode 100644 index 0000000..1d6364f --- /dev/null +++ b/src/klomanager.h @@ -0,0 +1,59 @@ +/*************************************************************************** +* Copyright (C) 2006 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef KLOMANAGER_H +#define KLOMANAGER_H + +#include <tqdom.h> +#include <tqfont.h> + +#include "datablocks/kreborder.h" + +class KLOManager +{ +public: + KLOManager(); + virtual ~KLOManager(); + + static TQStringList classes(); + +protected: + void processDocument( const TQDomDocument & ); + + virtual void loadBackgroundColor( const TQString &/*obj*/, const TQColor& ){}; + virtual void loadFont( const TQString &/*obj*/, const TQFont& ){}; + virtual void loadTextColor( const TQString &/*obj*/, const TQColor& ){}; + virtual void loadVisibility( const TQString &/*obj*/, bool ){}; + virtual void loadAlignment( const TQString &/*obj*/, int ){}; + virtual void loadBorder( const TQString &/*obj*/, const KreBorder& ){}; + virtual void loadColumns( const TQString &/*obj*/, int ){}; + + virtual void beginObject( const TQString &/*obj*/ ){}; + virtual void endObject(){}; + + bool getBoolAttribute( const TQDomElement &obj, const TQString &attr, bool defaultValue = true ) const; + TQColor getColorAttribute( const TQDomElement &obj, const TQString &attr, const TQColor &defaultValue = TQt::white ) const; + TQString getTextAttribute( const TQDomElement &obj, const TQString &attr, const TQString &defaultValue = TQString::null ) const; + int getIntAttribute( const TQDomElement &obj, const TQString &attr, int defaultValue = 0 ) const; + KreBorder getBorderAttribute( const TQDomElement &obj, const TQString &attr, const KreBorder &defaultValue = KreBorder() ) const; + TQFont getFontAttribute( const TQDomElement &obj, const TQString &attr, const TQFont &defaultValue = TQFont() ) const; + + TQString alignmentAsCSS( int ); + TQString borderAsCSS( const KreBorder & ); + TQString bgColorAsCSS( const TQColor & ); + TQString fontAsCSS( const TQFont & ); + TQString textColorAsCSS( const TQColor & ); + TQString visibilityAsCSS( bool ); + +private: + TQDomElement getLayoutAttribute( const TQDomElement &obj, const TQString &attr ) const; +}; + +#endif //KLOMANAGER_H diff --git a/src/krecipes.cpp b/src/krecipes.cpp new file mode 100644 index 0000000..fa0b284 --- /dev/null +++ b/src/krecipes.cpp @@ -0,0 +1,703 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* * +* Copyright (C) 2003-2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "pref.h" +#include "krecipes.h" +#include "krecipesview.h" + +#include "dialogs/recipeviewdialog.h" +#include "dialogs/recipeinputdialog.h" +#include "dialogs/selectrecipedialog.h" +#include "dialogs/ingredientsdialog.h" +#include "dialogs/propertiesdialog.h" +#include "dialogs/shoppinglistdialog.h" +#include "dialogs/categorieseditordialog.h" +#include "dialogs/authorsdialog.h" +#include "dialogs/unitsdialog.h" +#include "dialogs/ingredientmatcherdialog.h" +#include "dialogs/dbimportdialog.h" +#include "dialogs/pagesetupdialog.h" +#include "dialogs/recipeimportdialog.h" +#include "dialogs/similarcategoriesdialog.h" +#include "dialogs/conversiondialog.h" + +#include "importers/kreimporter.h" +#include "importers/mmfimporter.h" +#include "importers/mx2importer.h" +#include "importers/mxpimporter.h" +#include "importers/nycgenericimporter.h" +#include "importers/recipemlimporter.h" +#include "importers/rezkonvimporter.h" +#include "importers/kredbimporter.h" + +#include "datablocks/recipe.h" +#include "backends/recipedb.h" +#include "backends/progressinterface.h" + +#include <tqdragobject.h> +#include <kprinter.h> +#include <tqpainter.h> +#include <tqpaintdevicemetrics.h> +#include <tqmessagebox.h> + +#include <kprogress.h> +#include <tdemessagebox.h> +#include <tdeglobal.h> +#include <tdelocale.h> +#include <kiconloader.h> +#include <tdemenubar.h> +#include <kstatusbar.h> +#include <kkeydialog.h> +#include <tdeaccel.h> +#include <tdeio/netaccess.h> +#include <tdefiledialog.h> +#include <tdeconfig.h> +#include <kcursor.h> + +#include <kedittoolbar.h> +#include <tdestdaccel.h> +#include <tdeaction.h> +#include <kstdaction.h> +//Settings headers +#include <tdeversion.h> + +Krecipes::Krecipes() + : TDEMainWindow( 0, "Krecipes" ), + m_view( new KrecipesView( this ) ), + m_printer( 0 ) +{ + // accept dnd + setAcceptDrops( true ); + + // tell the TDEMainWindow that this is indeed the main widget + setCentralWidget( m_view ); + + // then, setup our actions + setupActions(); + + // and a status bar + statusBar() ->show(); + + // apply the saved mainwindow settings, if any, and ask the mainwindow + // to automatically save settings if changed: window size, toolbar + // position, icon size, etc. + setAutoSaveSettings(); + + + // allow the view to change the statusbar and caption + connect( m_view, TQ_SIGNAL( signalChangeStatusbar( const TQString& ) ), + this, TQ_SLOT( changeStatusbar( const TQString& ) ) ); + connect( m_view, TQ_SIGNAL( signalChangeCaption( const TQString& ) ), + this, TQ_SLOT( changeCaption( const TQString& ) ) ); + + connect( m_view, TQ_SIGNAL( panelShown( KrePanel, bool ) ), TQ_SLOT( updateActions( KrePanel, bool ) ) ); + connect( m_view, TQ_SIGNAL( panelShown( KrePanel, bool ) ), TQ_SLOT( updateActions( KrePanel, bool ) ) ); + + connect( m_view, TQ_SIGNAL( recipeSelected(bool) ), TQ_SLOT( recipeSelected(bool) ) ); + + + // Enable/Disable the Save Button (Initialize disabled, and connect signal) + + connect( m_view, TQ_SIGNAL( enableSaveOption( bool ) ), this, TQ_SLOT( enableSaveOption( bool ) ) ); + + enableSaveOption( false ); // Disables saving initially + recipeSelected( false ); //nothing is selected initially + + parsing_file_dlg = new KDialog( this, "parsing_file_dlg", true, TQt::WX11BypassWM ); + TQLabel *parsing_file_dlg_label = new TQLabel( i18n( "Gathering recipe data from file.\nPlease wait..." ), parsing_file_dlg ); + parsing_file_dlg_label->setFrameStyle( TQFrame::Box | TQFrame::Raised ); + ( new TQVBoxLayout( parsing_file_dlg ) ) ->addWidget( parsing_file_dlg_label ); + parsing_file_dlg->adjustSize(); + //parsing_file_dlg->setFixedSize(parsing_file_dlg->size()); + + convertDialog = new ConversionDialog(this,m_view->database); +} + +Krecipes::~Krecipes() +{} + +void Krecipes::updateActions( KrePanel panel, bool show ) +{ + switch ( panel ) { + case RecipeView: { + exportAction->setEnabled( show ); + printAction->setEnabled( show ); + reloadAction->setEnabled( show ); + copyToClipboardAction->setEnabled( show ); + + //can't edit when there are multiple recipes loaded + if ( show && m_view->viewPanel->recipesLoaded() == 1 ) { + editAction->setEnabled( true ); + } + else + editAction->setEnabled( false ); + + break; + } + case SelectP: { + exportAction->setEnabled( show ); + editAction->setEnabled( show ); + copyToClipboardAction->setEnabled( show ); + break; + } + default: + break; + } +} + +void Krecipes::recipeSelected( bool selected ) +{ + copyToClipboardAction->setEnabled( selected ); + editAction->setEnabled( selected ); +} + +void Krecipes::setupActions() +{ + printAction = KStdAction::print( this, TQ_SLOT( filePrint() ), actionCollection() ); + reloadAction = new TDEAction( i18n( "Reloa&d" ), "reload", Key_F5, m_view, TQ_SLOT( reloadDisplay() ), actionCollection(), "reload_action" ); + + editAction = new TDEAction( i18n( "&Edit Recipe" ), "edit", CTRL + Key_E, + m_view, TQ_SLOT( editRecipe() ), + actionCollection(), "edit_action" ); + + (void) new TDEAction( i18n( "&Measurement Converter" ), "", CTRL + Key_M, + this, TQ_SLOT( conversionToolSlot() ), + actionCollection(), "converter_action" ); + + TDEConfig *config = TDEGlobal::config(); + config->setGroup("Advanced"); + if ( config->readBoolEntry("UnhideMergeTools",false) ) { + ( void ) new TDEAction( i18n( "&Merge Similar Categories..." ), "categories", CTRL + Key_M, + this, TQ_SLOT( mergeSimilarCategories() ), + actionCollection(), "merge_categories_action" ); + + ( void ) new TDEAction( i18n( "&Merge Similar Ingredients..." ), "ingredients", CTRL + Key_M, + this, TQ_SLOT( mergeSimilarIngredients() ), + actionCollection(), "merge_ingredients_action" ); + } + + TDEAction *action = KStdAction::openNew( this, TQ_SLOT( fileNew() ), actionCollection() ); + action->setText( i18n( "&New Recipe" ) ); + + saveAction = KStdAction::save( this, TQ_SLOT( fileSave() ), actionCollection() ); + + KStdAction::quit( kapp, TQ_SLOT( quit() ), actionCollection() ); + + m_toolbarAction = KStdAction::showToolbar( this, TQ_SLOT( optionsShowToolbar() ), actionCollection() ); + m_statusbarAction = KStdAction::showStatusbar( this, TQ_SLOT( optionsShowStatusbar() ), actionCollection() ); + + KStdAction::keyBindings( this, TQ_SLOT( optionsConfigureKeys() ), actionCollection() ); + KStdAction::configureToolbars( this, TQ_SLOT( optionsConfigureToolbars() ), actionCollection() ); + KStdAction::preferences( this, TQ_SLOT( optionsPreferences() ), actionCollection() ); + + ( void ) new TDEAction( i18n( "Import from File..." ), CTRL + Key_I, + this, TQ_SLOT( import() ), + actionCollection(), "import_action" ); + + ( void ) new TDEAction( i18n( "Import from Database..." ), 0, + this, TQ_SLOT( kreDBImport() ), + actionCollection(), "import_db_action" ); + + exportAction = new TDEAction( i18n( "Export..." ), 0, + this, TQ_SLOT( fileExport() ), + actionCollection(), "export_action" ); + + copyToClipboardAction = new TDEAction( i18n( "&Copy to Clipboard" ), "edit-copy", + CTRL + Key_C, + this, TQ_SLOT( fileToClipboard() ), + actionCollection(), "copy_to_clipboard_action" ); + + ( void ) new TDEAction( i18n( "Page Setup..." ), 0, + this, TQ_SLOT( pageSetupSlot() ), + actionCollection(), "page_setup_action" ); + + ( void ) new TDEAction( i18n( "Print Setup..." ), 0, + this, TQ_SLOT( printSetupSlot() ), + actionCollection(), "print_setup_action" ); + + ( void ) new TDEAction( i18n( "Backup..." ), "krecipes_file", 0, + this, TQ_SLOT( backupSlot() ), + actionCollection(), "backup_action" ); + + ( void ) new TDEAction( i18n( "Restore..." ), 0, + this, TQ_SLOT( restoreSlot() ), + actionCollection(), "restore_action" ); + + updateActions( SelectP, true ); + updateActions( RecipeView, false ); + + createGUI(); +} + +void Krecipes::saveProperties( TDEConfig * ) +{ + // the 'config' object points to the session managed + // config file. anything you write here will be available + // later when this app is restored + + //if (!m_view->currentURL().isNull()) + // config->writeEntry("lastURL", m_view->currentURL()); +} + +void Krecipes::readProperties( TDEConfig * ) +{ + // the 'config' object points to the session managed + // config file. this function is automatically called whenever + // the app is being restored. read in here whatever you wrote + // in 'saveProperties' + + //TQString url = config->readEntry("lastURL"); + + //if (!url.isNull()) + // m_view->openURL(KURL(url)); +} + +void Krecipes::dragEnterEvent( TQDragEnterEvent *event ) +{ + // accept uri drops only + event->accept( TQUriDrag::canDecode( event ) ); +} + + +void Krecipes::fileNew() +{ + + // Create a new element (Element depends on active panel. New recipe by default) + m_view->createNewElement(); +} + +void Krecipes::fileOpen() +{ + // this slot is called whenever the File->Open menu is selected, + // the Open shortcut is pressed (usually CTRL+O) or the Open toolbar + // button is clicked + /* + // this brings up the generic open dialog + KURL url = KURLRequesterDlg::getURL(TQString::null, this, i18n("Open Location") ); + */ + // standard filedialog + /*KURL url = KFileDialog::getOpenURL(TQString::null, TQString::null, this, i18n("Open Location")); + if (!url.isEmpty()) + m_view->openURL(url);*/ +} + +void Krecipes::fileSave() +{ + // this slot is called whenever the File->Save menu is selected, + // the Save shortcut is pressed (usually CTRL+S) or the Save toolbar + // button is clicked + m_view->save(); +} + +void Krecipes::fileExport() +{ + // this slot is called whenever the File->Export menu is selected, + m_view->exportRecipe(); +} + +void Krecipes::fileToClipboard() +{ + m_view->exportToClipboard(); +} + +void Krecipes::filePrint() +{ + m_view->print(); +} + +void Krecipes::import() +{ + KFileDialog file_dialog( TQString::null, + "*.kre *.kreml|Krecipes (*.kre, *.kreml)\n" + "*.mx2|MasterCook (*.mx2)\n" + "*.mxp *.txt|MasterCook Export (*.mxp, *.txt)\n" + "*.mmf *.txt|Meal-Master (*.mmf, *.txt)\n" + "*.txt|\"Now You're Cooking\" Generic Export (*.txt)\n" + "*.xml *.recipeml|RecipeML (*.xml, *.recipeml)\n" + "*.rk *.txt|Rezkonv (*.rk, *.txt)", + this, + "file_dialog", + true + ); + file_dialog.setMode( KFile::Files ); + + if ( file_dialog.exec() == KFileDialog::Accepted ) { + TQStringList warnings_list; + + TQString selected_filter = file_dialog.currentFilter(); + + BaseImporter *importer; + if ( selected_filter == "*.mxp *.txt" ) + importer = new MXPImporter(); + else if ( selected_filter == "*.mmf *.txt" ) + importer = new MMFImporter(); + else if ( selected_filter == "*.txt" ) + importer = new NYCGenericImporter(); + else if ( selected_filter == "*.mx2" ) + importer = new MX2Importer(); + else if ( selected_filter == "*.kre *.kreml" ) + importer = new KreImporter(); + else if ( selected_filter == "*.xml *.recipeml" ) + importer = new RecipeMLImporter(); + else if ( selected_filter == "*.rk *.txt" ) + importer = new RezkonvImporter(); + else { + KMessageBox::sorry( this, + TQString( i18n( "Filter \"%1\" not recognized.\n" + "Please select one of the provided filters." ) ).arg( selected_filter ), + i18n( "Unrecognized Filter" ) + ); + import(); //let's try again :) + return ; + } + + parsing_file_dlg->show(); + TDEApplication::setOverrideCursor( KCursor::waitCursor() ); + importer->parseFiles( file_dialog.selectedFiles() ); + parsing_file_dlg->hide(); + TDEApplication::restoreOverrideCursor(); + + TDEConfig * config = kapp->config(); + config->setGroup( "Import" ); + bool direct = config->readBoolEntry( "DirectImport", false ); + if ( !direct ) { + RecipeImportDialog import_dialog( importer->recipeList() ); + + if ( import_dialog.exec() != TQDialog::Accepted ) { + delete importer; + return; + } + else + importer->setRecipeList( import_dialog.getSelectedRecipes() ); + } + + importer->import(m_view->database); + //m_view->database->import( importer ); //TODO TESTS: Do it this way + + if ( !importer->getMessages().isEmpty() ) { + KTextEdit * warningEdit = new KTextEdit( this ); + warningEdit->setTextFormat( TQt::RichText ); + warningEdit->setText( TQString( i18n( "NOTE: We recommend that all recipes generating warnings be checked to ensure that they were properly imported, and no loss of recipe data has occurred.<br><br>" ) ) + importer->getMessages() ); + warningEdit->setReadOnly( true ); + + KDialogBase showWarningsDlg( KDialogBase::Swallow, i18n( "Import Warnings" ), KDialogBase::Ok, KDialogBase::Default, this ); + showWarningsDlg.setMainWidget( warningEdit ); //KDialogBase will delete warningEdit for us + showWarningsDlg.resize( TQSize( 550, 250 ) ); + showWarningsDlg.exec(); + } + + delete importer; + } +} + +void Krecipes::kreDBImport() +{ + DBImportDialog importOptions; + if ( importOptions.exec() == TQDialog::Accepted ) { + //Get all params, even if we don't use them + TQString host, user, pass, table; + int port; + importOptions.serverParams( host, user, pass, port, table ); + + KreDBImporter importer( importOptions.dbType(), host, user, pass, port ); //last 4 params may or may not be even used (depends on dbType) + + parsing_file_dlg->show(); + TDEApplication::setOverrideCursor( KCursor::waitCursor() ); + TQStringList tables; + if ( importOptions.dbType() == "SQLite" ) + tables << importOptions.dbFile(); + else //MySQL or PostgreSQL + tables << table; + importer.parseFiles( tables ); + parsing_file_dlg->hide(); + TDEApplication::restoreOverrideCursor(); + + TDEConfig * config = TDEGlobal::config(); + config->setGroup( "Import" ); + bool direct = config->readBoolEntry( "DirectImport", false ); + if ( !direct ) { + RecipeImportDialog import_dialog( importer.recipeList() ); + + if ( import_dialog.exec() != TQDialog::Accepted ) { + return; + } + else + importer.setRecipeList( import_dialog.getSelectedRecipes() ); + } + + TQString error = importer.getErrorMsg(); + if ( !error.isEmpty() ) { + KMessageBox::sorry( this, error ); + } + else + importer.import(m_view->database); + } +} + +void Krecipes::pageSetupSlot() +{ + Recipe recipe; + m_view->selectPanel->getCurrentRecipe( &recipe ); + + TDEConfig *config = TDEGlobal::config(); + config->setGroup("Page Setup"); + TQString layout = config->readEntry( "Layout", locate( "appdata", "layouts/Default.klo" ) ); + TQString printLayout = config->readEntry( "PrintLayout", locate( "appdata", "layouts/Default.klo" ) ); + + if ( layout == printLayout ) { + KMessageBox::information( this, i18n("The recipe print and view layouts use the same file for their style, meaning changing one view's look changes them both. If this is not the behavior you desire, load one style and save it under a different name."), + TQString::null, "sharedLayoutWarning" ); + } + + PageSetupDialog page_setup( this, recipe ); + page_setup.exec(); +} + +void Krecipes::printSetupSlot() +{ + Recipe recipe; + m_view->selectPanel->getCurrentRecipe( &recipe ); + + TDEConfig *config = TDEGlobal::config(); + config->setGroup("Page Setup"); + TQString layout = config->readEntry( "Layout", locate( "appdata", "layouts/Default.klo" ) ); + TQString printLayout = config->readEntry( "PrintLayout", locate( "appdata", "layouts/Default.klo" ) ); + + if ( layout == printLayout ) { + KMessageBox::information( this, i18n("The recipe print and view layouts use the same file for their style, meaning changing one view's look changes them both. If this is not the behavior you desire, load one style and save it under a different name."), + TQString::null, "sharedLayoutWarning" ); + } + + PageSetupDialog pageSetup( this, recipe, "Print" ); + pageSetup.exec(); +} + +void Krecipes::conversionToolSlot() +{ + convertDialog->show(); +} + +void Krecipes::backupSlot() +{ + TQString fileName = KFileDialog::getSaveFileName(TQString::null, + TQString("*.krecbk|%1 (*.krecbk)").arg("Krecipes Backup File"), + this,i18n("Save Backup As...")); + + int overwrite = KMessageBox::Yes; + if ( TQFile::exists(fileName) ) { + overwrite = KMessageBox::warningYesNo( this, TQString( i18n( "File \"%1\" exists. Are you sure you want to overwrite it?" ) ).arg( fileName ) ); + } + + if ( !fileName.isNull() && overwrite == KMessageBox::Yes ) { + ProgressInterface pi(this); + pi.listenOn(m_view->database); + + TQString errMsg; + if ( !m_view->database->backup( fileName, &errMsg ) ) + KMessageBox::error( this, errMsg, i18n("Backup Failed") ); + } +} + +void Krecipes::restoreSlot() +{ + TQString filename = KFileDialog::getOpenFileName(TQString::null, + TQString("*.krecbk|%1 (*.krecbk)").arg(i18n("Krecipes Backup File")), + this,i18n("Restore Backup")); + + if ( !filename.isNull() ) { + switch ( KMessageBox::warningContinueCancel(this,i18n("<b>Restoring this file will erase ALL data currently in the database!</b><br /><br />If you want to keep the recipes in your database, click \"Cancel\" and first export your recipes. These can then be imported once the restore is complete.<br /><br />Are you sure you want to proceed?"),TQString::null,KStdGuiItem::cont(),"RestoreWarning") ) { + case KMessageBox::Continue: { + ProgressInterface pi(this); + pi.listenOn(m_view->database); + + TQString errMsg; + if ( m_view->database->restore( filename, &errMsg ) ) + KMessageBox::information(this,i18n("Restore successful.")); + else + KMessageBox::error( this, errMsg, i18n("Restore Failed") ); + + pi.listenOn(0); + m_view->reload(); + } + case KMessageBox::Cancel: + default: break; + } + } +} + +void Krecipes::mergeSimilarCategories() +{ + ElementList categories; + m_view->database->loadCategories(&categories); + SimilarCategoriesDialog dlg(categories,this); + if ( dlg.exec() == TQDialog::Accepted ) { + TQValueList<int> ids = dlg.matches(); + TQString name = dlg.element(); + + int id = m_view->database->findExistingCategoryByName(name); + if ( id == -1 ) { + m_view->database->createNewCategory(name); + id = m_view->database->lastInsertID(); + } + + for ( TQValueList<int>::const_iterator it = ids.begin(); it != ids.end(); ++it ) { + if ( id != *it ) + m_view->database->mergeCategories(id, *it); + } + } +} + +void Krecipes::mergeSimilarIngredients() +{ + ElementList ingredients; + m_view->database->loadIngredients(&ingredients); + SimilarCategoriesDialog dlg(ingredients,this); + if ( dlg.exec() == TQDialog::Accepted ) { + TQValueList<int> ids = dlg.matches(); + TQString name = dlg.element(); + + if ( ids.isEmpty() || name.isEmpty() ) return; + + int id = m_view->database->findExistingIngredientByName(name); + if ( id == -1 ) { + m_view->database->createNewIngredient(name); + id = m_view->database->lastInsertID(); + } + + for ( TQValueList<int>::const_iterator it = ids.begin(); it != ids.end(); ++it ) { + if ( id != *it ) + m_view->database->mergeIngredients(id, *it); + } + } +} + +//return true to close app +bool Krecipes::queryClose() +{ + if ( !m_view->inputPanel->everythingSaved() ) { + switch ( KMessageBox::questionYesNoCancel( this, + i18n( "A recipe contains unsaved changes.\n" + "Do you want to save the changes before exiting?" ), + i18n( "Unsaved Changes" ) ) ) { + case KMessageBox::Yes: + return m_view->save(); + case KMessageBox::No: + return true; + case KMessageBox::Cancel: + return false; + default: + return true; + } + } + else + return true; +} + +void Krecipes::optionsShowToolbar() +{ + // this is all very cut and paste code for showing/hiding the + // toolbar + if ( m_toolbarAction->isChecked() ) + toolBar() ->show(); + else + toolBar() ->hide(); +} + +void Krecipes::optionsShowStatusbar() +{ + // this is all very cut and paste code for showing/hiding the + // statusbar + if ( m_statusbarAction->isChecked() ) + statusBar() ->show(); + else + statusBar() ->hide(); +} + +void Krecipes::optionsConfigureKeys() +{ +#if KDE_IS_VERSION(3,1,92 ) + // for KDE 3.2: KKeyDialog::configureKeys is deprecated + KKeyDialog::configure( actionCollection(), this, true ); +#else + + KKeyDialog::configureKeys( actionCollection(), "krecipesui.rc" ); +#endif + +} + +void Krecipes::optionsConfigureToolbars() +{ + // use the standard toolbar editor +#if defined(TDE_MAKE_VERSION) +# if TDE_VERSION >= TDE_MAKE_VERSION(3,1,0) + saveMainWindowSettings( TDEGlobal::config(), autoSaveGroup() ); +# else + + saveMainWindowSettings( TDEGlobal::config() ); +# endif +#else + + saveMainWindowSettings( TDEGlobal::config() ); +#endif + + KEditToolbar dlg( actionCollection() ); + connect( &dlg, TQ_SIGNAL( newToolbarConfig() ), this, TQ_SLOT( newToolbarConfig() ) ); + dlg.exec(); +} + +void Krecipes::newToolbarConfig() +{ + // this slot is called when user clicks "Ok" or "Apply" in the toolbar editor. + // recreate our GUI, and re-apply the settings (e.g. "text under icons", etc.) + createGUI(); + +#if defined(TDE_MAKE_VERSION) +# if TDE_VERSION >= TDE_MAKE_VERSION(3,1,0) + + applyMainWindowSettings( TDEGlobal::config(), autoSaveGroup() ); +# else + + applyMainWindowSettings( TDEGlobal::config() ); +# endif +#else + + applyMainWindowSettings( TDEGlobal::config() ); +#endif +} + +void Krecipes::optionsPreferences() +{ + + // popup some sort of preference dialog, here + KrecipesPreferences dlg( this ); + if ( dlg.exec() ) {} + +} + +void Krecipes::changeStatusbar( const TQString& text ) +{ + // display the text on the statusbar + statusBar() ->message( text ); +} + +void Krecipes::changeCaption( const TQString& text ) +{ + // display the text on the caption + setCaption( text ); +} +void Krecipes::enableSaveOption( bool en ) +{ + saveAction->setEnabled( en ); +} + +#include "krecipes.moc" diff --git a/src/krecipes.desktop b/src/krecipes.desktop new file mode 100644 index 0000000..97fafad --- /dev/null +++ b/src/krecipes.desktop @@ -0,0 +1,66 @@ +[Desktop Entry] +Name=KRecipes +Name[sv]=KRecept +Name[ta]=கிளிக்கர் +Name[xx]=xxKRecipesxx +Exec=krecipes %i %m -caption "%c" +Icon=krecipes +Type=Application +X-DocPath=krecipes/index.html +GenericName=Cooking Book +GenericName[bg]=Готварска книга +GenericName[br]=Levr keginer +GenericName[cs]=Kuchařka +GenericName[da]=Kogebog +GenericName[de]=Kochbuch +GenericName[el]=Βιβλίο μαγειρικής +GenericName[es]=Libro de cocina +GenericName[et]=Kokaraamat +GenericName[ga]=Leabhar Cócaireachta +GenericName[gl]=Libro de Receitas +GenericName[it]=Libro di ricette +GenericName[ja]=料理本 +GenericName[ka]=სამზარეულო წიგნი +GenericName[nb]=Kokebok +GenericName[nl]=Kookboek +GenericName[pa]=ਖਾਣਾ ਬਣਾਉਦ ਦੀ ਕਿਤਾਬ +GenericName[pt]=Livro de Culinária +GenericName[pt_BR]=Livro de Culinária +GenericName[sr]=Кувар +GenericName[sr@Latn]=Kuvar +GenericName[sv]=Kokbok +GenericName[uk]=Куховарська книга +GenericName[xx]=xxCooking Bookxx +GenericName[zh_CN]=食谱 +Comment=The TDE Cooking Book +Comment[bg]=Готварската книга на TDE +Comment[br]=Al levr keginer TDE +Comment[bs]=TDE kuhar +Comment[cs]=Kuchařka pro TDE +Comment[da]=TDE's kogebog +Comment[de]=Das TDE Kochbuch +Comment[el]=Το βιβλίο μαγειρικής του TDE +Comment[es]=El libro de cocina de TDE +Comment[et]=TDE kokaraamat +Comment[fr]=Le guide de TDE +Comment[ga]=Leabhar Cócaireachta TDE +Comment[gl]=O Libro de Receitas de TDE +Comment[it]=Il libro delle ricette di TDE +Comment[ja]=TDE 料理本 +Comment[ka]=TDE-ს სამზარეულო წიგნი +Comment[nb]=TDE kokebok +Comment[nl]=Het TDE-kookboek +Comment[pa]=TDE ਖਾਣਾ ਬਣਾਉਣ ਦੀ ਕਿਤਾਬ +Comment[pt]=Livro de Culinária do TDE +Comment[pt_BR]=Livro de Culinária do TDE +Comment[ru]=Книга рецептов +Comment[sr]=Кувар за TDE +Comment[sr@Latn]=Kuvar za TDE +Comment[sv]=TDE:s kokbok +Comment[ta]=TDE சமைக்குக்ம் புத்தகம் +Comment[uk]=Куховарська книга для TDE +Comment[xx]=xxThe TDE Cooking Bookxx +Comment[zh_CN]=TDE 食谱 +Terminal=false +X-DCOP-ServiceType=Unique +Categories=Utility;Database;Qt;TDE; diff --git a/src/krecipes.h b/src/krecipes.h new file mode 100644 index 0000000..c689e19 --- /dev/null +++ b/src/krecipes.h @@ -0,0 +1,140 @@ +/*************************************************************************** +* Copyright (C) 2003-2004 by * +* Unai Garro ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef _KRECIPES_H_ +#define _KRECIPES_H_ + +#ifdef HAVE_CONFIG_H +#include <config.h> +#endif + +#include <tdeapplication.h> +#include <tdemainwindow.h> + +#include "krecipesview.h" //for KrePanel + +class KrecipesView; + +class KPrinter; +class TDEToggleAction; +class KURL; +class KDialog; +class ConversionDialog; + +/** + * This class serves as the main window for Krecipes. It handles the + * menus, toolbars, and status bars. + * + * @short Main window class + * @author $AUTHOR <$EMAIL> + * @version $APP_VERSION + */ +class Krecipes : public TDEMainWindow +{ + TQ_OBJECT +public: + /** + * Default Constructor + */ + Krecipes(); + + /** + * Default Destructor + */ + virtual ~Krecipes(); + + +protected: + /** + * Overridden virtuals for TQt drag 'n drop (XDND) + */ + virtual void dragEnterEvent( TQDragEnterEvent *event ); + +protected: + /** + * This function is called when it is time for the app to save its + * properties for session management purposes. + */ + void saveProperties( TDEConfig * ); + + /** + * This function is called when this app is restored. The TDEConfig + * object points to the session management config file that was saved + * with @ref saveProperties + */ + void readProperties( TDEConfig * ); + + virtual bool queryClose(); + + +private slots: + void fileNew(); + void fileOpen(); + void fileSave(); + void fileExport(); + void fileToClipboard(); + void filePrint(); + void optionsShowToolbar(); + void optionsShowStatusbar(); + void optionsConfigureKeys(); + void optionsConfigureToolbars(); + void optionsPreferences(); + void newToolbarConfig(); + void import(); + void kreDBImport(); + void pageSetupSlot(); + void printSetupSlot(); + void conversionToolSlot(); + void backupSlot(); + void restoreSlot(); + void mergeSimilarCategories(); + void mergeSimilarIngredients(); + + /** This function is called whenever a panel is shown or hidden and then sets + * actions as enabled as appropriate. + */ + void updateActions( KrePanel panel, bool show ); + + void changeStatusbar( const TQString& text ); + void changeCaption( const TQString& text ); + +private: + // Private methods + void setupAccel(); + void setupActions(); + +private: + KrecipesView *m_view; + + KPrinter *m_printer; + TDEToggleAction *m_toolbarAction; + TDEToggleAction *m_statusbarAction; + +private: + // Internal variables + TDEAction *saveAction; + TDEAction *exportAction; + TDEAction *editAction; + TDEAction *printAction; + TDEAction *reloadAction; + TDEAction *copyToClipboardAction; + + KDialog *parsing_file_dlg; + ConversionDialog *convertDialog; + + TQValueList<TDEAction*> recipe_actions; + +private slots: + void enableSaveOption( bool en = true ); + void recipeSelected( bool ); +}; + +#endif // _KRECIPES_H_ diff --git a/src/krecipesdbiface.h b/src/krecipesdbiface.h new file mode 100644 index 0000000..f5447dc --- /dev/null +++ b/src/krecipesdbiface.h @@ -0,0 +1,26 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef _KRECIPESDBIFACE_H_ +#define _KRECIPESDBIFACE_H_ + +#include <dcopobject.h> + +class KrecipesDBIface : virtual public DCOPObject +{ + K_DCOP +public: + +k_dcop: + virtual void emptyData() = 0; + virtual bool backup( const TQString &filename ) = 0; +}; + +#endif // _KRECIPESIFACE_H_ diff --git a/src/krecipesiface.h b/src/krecipesiface.h new file mode 100644 index 0000000..fe9173f --- /dev/null +++ b/src/krecipesiface.h @@ -0,0 +1,31 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef _KRECIPESIFACE_H_ +#define _KRECIPESIFACE_H_ + +#include <dcopobject.h> +#include <dcopref.h> + +#include <tqvaluelist.h> + +class KrecipesIface : virtual public DCOPObject +{ + K_DCOP +public: + +k_dcop: + virtual DCOPRef currentDatabase() const = 0; + virtual void reload() = 0; + + virtual void exportRecipes( const TQValueList<int> &ids ) = 0; +}; + +#endif // _KRECIPESIFACE_H_ diff --git a/src/krecipesui.rc b/src/krecipesui.rc new file mode 100644 index 0000000..6faecfe --- /dev/null +++ b/src/krecipesui.rc @@ -0,0 +1,27 @@ +<!DOCTYPE kpartgui SYSTEM "kpartgui.dtd"> +<kpartgui name="krecipes" version="10"> +<MenuBar> + <Menu name="file"><text>&File</text> + <Action name="reload_action" /> + <Separator /> + <Action name="import_action" /> + <Action name="import_db_action" /> + <Action name="export_action" /> + <Separator /> + <Action name="backup_action" /> + <Action name="restore_action" /> + </Menu> + <Menu name="edit"><text>&Edit</text> + <Action name="page_setup_action" /> + <Action name="print_setup_action" /> + <Separator /> + <Action name="copy_to_clipboard_action" /> +</Menu> + <Menu name="tools"><text>&Tools</text> + <Action name="edit_action" /> + <Action name="converter_action" /> + <Action name="merge_categories_action" /> + <Action name="merge_ingredients_action" /> +</Menu> +</MenuBar> +</kpartgui> diff --git a/src/krecipesview.cpp b/src/krecipesview.cpp new file mode 100644 index 0000000..b9f79d1 --- /dev/null +++ b/src/krecipesview.cpp @@ -0,0 +1,942 @@ + +/*************************************************************************** +* Copyright (C) 2003-2004 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "krecipesview.h" + +#include <cstdlib> + +#include <tqlayout.h> +#include <tqimage.h> +#include <tqpainter.h> +#include <tqpalette.h> + +#include <tdeapplication.h> +#include <tdeconfig.h> +#include <kdebug.h> +#include <tdeglobalsettings.h> +#include <klibloader.h> +#include <tdelocale.h> +#include <tdemessagebox.h> +#include <kprogress.h> +#include <krun.h> +#include <ktrader.h> +#include <kurl.h> +#include <kcursor.h> + +#include "recipeactionshandler.h" +#include "setupwizard.h" +#include "kstartuplogo.h" + +#include "dialogs/recipeinputdialog.h" +#include "dialogs/recipeviewdialog.h" +#include "dialogs/selectrecipedialog.h" +#include "dialogs/ingredientsdialog.h" +#include "dialogs/propertiesdialog.h" +#include "dialogs/shoppinglistdialog.h" +#include "dialogs/dietwizarddialog.h" +#include "dialogs/categorieseditordialog.h" +#include "dialogs/authorsdialog.h" +#include "dialogs/unitsdialog.h" +#include "dialogs/prepmethodsdialog.h" +#include "dialogs/ingredientmatcherdialog.h" + +#include "widgets/kremenu.h" +#include "widgets/paneldeco.h" + +#include "backends/progressinterface.h" + +#include "profiling.h" + +KrecipesView::KrecipesView( TQWidget *parent ) + : DCOPObject( "KrecipesInterface" ), TQVBox( parent ) +{ + #ifndef NDEBUG + TQTime dbg_total_timer; dbg_total_timer.start(); + #endif + + kapp->dcopClient()->setDefaultObject( objId() ); + + // Init the setup wizard if necessary + kdDebug() << "Beginning wizard" << endl; + wizard(); + kdDebug() << "Wizard finished correctly" << endl; + + // Show Splash Screen + + TDEStartupLogo* start_logo = 0L; + start_logo = new TDEStartupLogo(); + start_logo -> setHideEnabled( true ); + start_logo->show(); + start_logo->raise(); + + // Initialize Database + + // Read the database setup + + TDEConfig *config; + config = kapp->config(); + config->sync(); + + + // Check if the database type is among those supported + // and initialize the database in each case + START_TIMER("Initializing database") + initDatabase( config ); + END_TIMER() + + + // Design the GUI + splitter = new TQHBox( this ); + + // Create Left and Right Panels (splitter) + + + TDEIconLoader il; + leftPanel = new KreMenu( splitter, "leftPanel" ); + rightPanel = new PanelDeco( splitter, "rightPanel", i18n( "Find/Edit Recipes" ), "filefind" ); + + // Design Left Panel + + START_TIMER("Setting up buttons") + // Buttons + buttonsList = new TQPtrList<KreMenuButton>(); + buttonsList->setAutoDelete( TRUE ); + + button0 = new KreMenuButton( leftPanel, SelectP ); + button0->setIconSet( il.loadIconSet( "filefind", TDEIcon::Panel, 32 ) ); + buttonsList->append( button0 ); + + button1 = new KreMenuButton( leftPanel, ShoppingP ); + button1->setIconSet( il.loadIconSet( "trolley", TDEIcon::Panel, 32 ) ); + buttonsList->append( button1 ); + + button7 = new KreMenuButton( leftPanel, DietP ); + button7->setIconSet( il.loadIconSet( "diet", TDEIcon::Panel, 32 ) ); + buttonsList->append( button7 ); + + button8 = new KreMenuButton( leftPanel, MatcherP ); + button8->setIconSet( il.loadIconSet( "categories", TDEIcon::Panel, 32 ) ); + buttonsList->append( button8 ); + + + // Submenus + dataMenu = leftPanel->createSubMenu( i18n( "Data" ), "2rightarrow" ); + + button2 = new KreMenuButton( leftPanel, IngredientsP, dataMenu ); + button2->setIconSet( il.loadIconSet( "ingredients", TDEIcon::Panel, 32 ) ); + //buttonsList->append(button2); + + button3 = new KreMenuButton( leftPanel, PropertiesP, dataMenu ); + button3->setIconSet( il.loadIconSet( "properties", TDEIcon::Panel, 32 ) ); + buttonsList->append( button3 ); + + button4 = new KreMenuButton( leftPanel, UnitsP, dataMenu ); + button4->setIconSet( il.loadIconSet( "units", TDEIcon::Panel, 32 ) ); + buttonsList->append( button4 ); + + button9 = new KreMenuButton( leftPanel, PrepMethodsP, dataMenu ); + button9->setIconSet( il.loadIconSet( "ICON PLEASE", TDEIcon::Panel, 32 ) ); + buttonsList->append( button9 ); + + button5 = new KreMenuButton( leftPanel, CategoriesP, dataMenu ); + button5->setIconSet( il.loadIconSet( "categories", TDEIcon::Panel, 32 ) ); + buttonsList->append( button5 ); + + button6 = new KreMenuButton( leftPanel, AuthorsP, dataMenu ); + button6->setIconSet( il.loadIconSet( "preferences-desktop-personal", TDEIcon::Panel, 32 ) ); + buttonsList->append( button6 ); + + contextButton = new TQPushButton( leftPanel, "contextButton" ); + contextButton->setIconSet( il.loadIconSet( "krectip", TDEIcon::Panel, 32 ) ); + contextButton->setGeometry( leftPanel->width() - 42, leftPanel->height() - 42, 32, 32 ); + contextButton->setPaletteBackgroundColor( contextButton->paletteBackgroundColor().light( 140 ) ); + contextButton->setFlat( true ); + END_TIMER() + + config->setGroup( "Performance" ); + int limit = config->readNumEntry( "CategoryLimit", -1 ); + database->updateCategoryCache(limit); + + // Right Panel Widgets + START_TIMER("Creating input dialog") + inputPanel = new RecipeInputDialog( rightPanel, database ); + END_TIMER() + + START_TIMER("Creating recipe view") + viewPanel = new RecipeViewDialog( rightPanel, database ); + END_TIMER() + + START_TIMER("Creating recipe selection dialog") + selectPanel = new SelectRecipeDialog( rightPanel, database ); + END_TIMER() + + START_TIMER("Creating ingredients component") + ingredientsPanel = new IngredientsDialog( rightPanel, database ); + END_TIMER() + + START_TIMER("Creating properties component") + propertiesPanel = new PropertiesDialog( rightPanel, database ); + END_TIMER() + + START_TIMER("Creating units component") + unitsPanel = new UnitsDialog( rightPanel, database ); + END_TIMER() + + START_TIMER("Creating shopping list dialog") + shoppingListPanel = new ShoppingListDialog( rightPanel, database ); + END_TIMER() + + START_TIMER("Creating diet wizard dialog") + dietPanel = new DietWizardDialog( rightPanel, database ); + END_TIMER() + + START_TIMER("Creating categories component") + categoriesPanel = new CategoriesEditorDialog( rightPanel, database ); + END_TIMER() + + START_TIMER("Creating authors component") + authorsPanel = new AuthorsDialog( rightPanel, database ); + END_TIMER() + + START_TIMER("Creating prep methods component") + prepMethodsPanel = new PrepMethodsDialog( rightPanel, database ); + END_TIMER() + + START_TIMER("Creating ingredients matcher dialog") + ingredientMatcherPanel = new IngredientMatcherDialog( rightPanel, database ); + END_TIMER() + + database->clearCategoryCache(); + + // Use to keep track of the panels + panelMap.insert( inputPanel, RecipeEdit ); + panelMap.insert( viewPanel, RecipeView ); + panelMap.insert( selectPanel, SelectP ); + panelMap.insert( ingredientsPanel, IngredientsP ); + panelMap.insert( propertiesPanel, PropertiesP ); + panelMap.insert( unitsPanel, UnitsP ); + panelMap.insert( shoppingListPanel, ShoppingP ); + panelMap.insert( dietPanel, DietP ); + panelMap.insert( categoriesPanel, CategoriesP ); + panelMap.insert( authorsPanel, AuthorsP ); + panelMap.insert( prepMethodsPanel, PrepMethodsP ); + panelMap.insert( ingredientMatcherPanel, MatcherP ); + + m_activePanel = RecipeEdit; + + // i18n + translate(); + + // Initialize Variables + recipeButton = 0; + + + + // Connect Signals from Left Panel to slotSetPanel() + connect( leftPanel, TQ_SIGNAL( clicked( KrePanel ) ), this, TQ_SLOT( slotSetPanel( KrePanel ) ) ); + + connect( contextButton, TQ_SIGNAL( clicked() ), TQ_SLOT( activateContextHelp() ) ); + + connect( leftPanel, TQ_SIGNAL( resized( int, int ) ), this, TQ_SLOT( resizeRightPane( int, int ) ) ); + + + // Retransmit signal to parent to Enable/Disable the Save Button + connect ( inputPanel, TQ_SIGNAL( enableSaveOption( bool ) ), this, TQ_SIGNAL( enableSaveOption( bool ) ) ); + + // Create a new button when a recipe is unsaved + connect ( inputPanel, TQ_SIGNAL( createButton( TQWidget*, const TQString & ) ), this, TQ_SLOT( addRecipeButton( TQWidget*, const TQString & ) ) ); + + // Connect Signals from selectPanel (SelectRecipeDialog) + + connect ( selectPanel, TQ_SIGNAL( recipeSelected( int, int ) ), this, TQ_SLOT( actionRecipe( int, int ) ) ); + connect ( selectPanel, TQ_SIGNAL( recipesSelected( const TQValueList<int>&, int ) ), this, TQ_SLOT( actionRecipes( const TQValueList<int>&, int ) ) ); + + // Connect Signals from ingredientMatcherPanel (IngredientMatcherDialog) + + connect ( ingredientMatcherPanel, TQ_SIGNAL( recipeSelected( int, int ) ), TQ_SLOT( actionRecipe( int, int ) ) ); + + // Close a recipe when requested (just switch panels) + connect( inputPanel, TQ_SIGNAL( closeRecipe() ), this, TQ_SLOT( closeRecipe() ) ); + + // Show a recipe when requested (just switch panels) + connect( inputPanel, TQ_SIGNAL( showRecipe( int ) ), this, TQ_SLOT( showRecipe( int ) ) ); + + // Create a new shopping list when a new diet is generated and accepted + connect( dietPanel, TQ_SIGNAL( dietReady() ), this, TQ_SLOT( createShoppingListFromDiet() ) ); + + // Place the Tip Button in correct position when the left pane is resized + connect( leftPanel, TQ_SIGNAL( resized( int, int ) ), this, TQ_SLOT( moveTipButton( int, int ) ) ); + + connect( rightPanel, TQ_SIGNAL( panelRaised( TQWidget*, TQWidget* ) ), TQ_SLOT( panelRaised( TQWidget*, TQWidget* ) ) ); + + connect( selectPanel, TQ_SIGNAL( recipeSelected(bool) ), TQ_SIGNAL( recipeSelected(bool) ) ); + + // Close Splash Screen + delete start_logo; + + #ifndef NDEBUG + kdDebug()<<"Total time elapsed: "<<dbg_total_timer.elapsed()/1000<<" sec"<<endl; + #endif +} + +KrecipesView::~KrecipesView() +{ + if (buttonsList) + delete buttonsList; + if (viewPanel) + delete viewPanel; //manually delete viewPanel because we need to be sure it is deleted + //before the database is because its destructor uses 'database' + if (database) + delete database; +} + +bool KrecipesView::questionRerunWizard( const TQString &message, const TQString &error ) +{ + TQString yesNoMessage = message + " " + i18n( "\nWould you like to run the setup wizard again? Otherwise, the application will be closed." ); + int answer = KMessageBox::questionYesNo( this, yesNoMessage ); + + if ( answer == KMessageBox::Yes ) + wizard( true ); + else { + kdError() << error << ". " << i18n( "Exiting" ) << endl; + kapp->exit( 1 ); exit ( 1 ); //FIXME: why doesn't kapp->exit(1) do anything? + return false; + } + + return true; +} + +void KrecipesView::translate() +{ + button0->setTitle( i18n( "Find/Edit Recipes" ) ); + button1->setTitle( i18n( "Shopping List" ) ); + button2->setTitle( i18n( "Ingredients" ) ); + button3->setTitle( i18n( "Properties" ) ); + button4->setTitle( i18n( "Units" ) ); + button9->setTitle( i18n( "Preparation Methods" ) ); + button5->setTitle( i18n( "Categories" ) ); + button6->setTitle( i18n( "Authors" ) ); + button7->setTitle( i18n( "Diet Helper" ) ); + button8->setTitle( i18n( "Ingredient Matcher" ) ); +} + +void KrecipesView::print() +{ + viewPanel->print(); +} + + +void KrecipesView::slotSetTitle( const TQString& title ) +{ + emit signalChangeCaption( title ); +} + +// Function to switch panels +void KrecipesView::slotSetPanel( KrePanel p ) +{ + m_activePanel = p; + + switch ( m_activePanel ) { + case SelectP: + rightPanel->setHeader( i18n( "Find/Edit Recipes" ), "filefind" ); + rightPanel->raise( selectPanel ); + break; + case ShoppingP: + rightPanel->setHeader( i18n( "Shopping List" ), "trolley" ); + rightPanel->raise( shoppingListPanel ); + shoppingListPanel->reload( Load ); + break; + case DietP: + rightPanel->setHeader( i18n( "Diet Helper" ), "diet" ); + rightPanel->raise( dietPanel ); + dietPanel->reload( Load ); + break; + case MatcherP: + rightPanel->setHeader( i18n( "Ingredient Matcher" ), "categories" ); + rightPanel->raise( ingredientMatcherPanel ); + ingredientMatcherPanel->reload( Load ); + break; + + case IngredientsP: + rightPanel->setHeader( i18n( "Ingredients" ), "ingredients" ); + rightPanel->raise( ingredientsPanel ); + ingredientsPanel->reload( Load ); + break; + case PropertiesP: + rightPanel->setHeader( i18n( "Properties" ), "properties" ); + rightPanel->raise( propertiesPanel ); + //propertiesPanel->reload(); + break; + case UnitsP: + rightPanel->setHeader( i18n( "Units" ), "units" ); + rightPanel->raise( unitsPanel ); + unitsPanel->reload( Load ); + break; + case PrepMethodsP: + rightPanel->setHeader( i18n( "Preparation Methods" ), "GIVE ME AN ICON :p" ); + rightPanel->raise( prepMethodsPanel ); + prepMethodsPanel->reload( Load ); + break; + case CategoriesP: + rightPanel->setHeader( i18n( "Categories" ), "categories" ); + rightPanel->raise( categoriesPanel ); + categoriesPanel->reload( Load ); + break; + case AuthorsP: + rightPanel->setHeader( i18n( "Authors" ), "preferences-desktop-personal" ); + rightPanel->raise( authorsPanel ); + authorsPanel->reload( Load ); + break; + case RecipeEdit: + rightPanel->setHeader( i18n( "Edit Recipe" ), "edit" ); + rightPanel->raise( inputPanel ); + break; + case RecipeView: + rightPanel->setHeader( i18n( "View Recipe" ), "filefind" ); + rightPanel->raise( viewPanel ); + break; + } +} + +bool KrecipesView::save( void ) +{ + return inputPanel->save(); +} + +/*! + \fn KrecipesView::exportRecipe() + */ +void KrecipesView::exportRecipe() +{ + TQWidget * vis_panel = rightPanel->visiblePanel(); + if ( vis_panel == viewPanel && viewPanel->recipesLoaded() > 0 ) { + exportRecipes( viewPanel->currentRecipes() ); + } + else if ( vis_panel == selectPanel ) { + selectPanel->getActionsHandler()->recipeExport(); + } +} + +void KrecipesView::exportToClipboard() +{ + TQWidget * vis_panel = rightPanel->visiblePanel(); + if ( vis_panel == viewPanel && viewPanel->recipesLoaded() > 0 ) { + TQValueList<int> ids = viewPanel->currentRecipes(); + RecipeActionsHandler::recipesToClipboard( ids, database ); + } + else if ( vis_panel == selectPanel ) { + selectPanel->getActionsHandler()->recipesToClipboard(); + } +} + +void KrecipesView::exportRecipes( const TQValueList<int> &ids ) +{ + if ( ids.count() == 1 ) + RecipeActionsHandler::exportRecipes( ids, i18n( "Export Recipe" ), database->recipeTitle( ids[ 0 ] ), database ); + else + RecipeActionsHandler::exportRecipes( ids, i18n( "Export Recipe" ), i18n( "Recipes" ), database ); +} + +void KrecipesView::actionRecipe( int recipeID, int action ) +{ + switch ( action ) { + case 0: //Show + { + showRecipe( recipeID ); + break; + } + case 1: // Edit + { + if ( !inputPanel->everythingSaved() ) + { + switch ( KMessageBox::questionYesNoCancel( this, + TQString( i18n( "A recipe contains unsaved changes.\n" + "Do you want to save changes made to this recipe before editing another recipe?" ) ), + i18n( "Unsaved changes" ) ) ) { + case KMessageBox::Yes: + inputPanel->save(); + break; + case KMessageBox::No: + break; + case KMessageBox::Cancel: + return ; + } + } + + inputPanel->loadRecipe( recipeID ); + slotSetPanel( RecipeEdit ); + break; + } + case 2: //Remove + { + switch ( KMessageBox::questionYesNo( this, + TQString( i18n( "Are you sure you want to permanently remove the recipe, %1?" ) ).arg(database->recipeTitle(recipeID)), + i18n( "Confirm remove" ) ) ) + { + case KMessageBox::Yes: + database->removeRecipe( recipeID ); + break; + case KMessageBox::No: + break; + } + break; + } + case 3: //Add to shopping list + { + shoppingListPanel->addRecipeToShoppingList( recipeID ); + break; + } + } +} + +void KrecipesView::actionRecipes( const TQValueList<int> &ids, int action ) +{ + if ( action == 0 ) //show + { + showRecipes( ids ); + } +} + + +void KrecipesView::createNewRecipe( void ) +{ + if ( !inputPanel->everythingSaved() ) { + switch ( KMessageBox::questionYesNoCancel( this, + TQString( i18n( "A recipe contains unsaved changes.\n" + "Do you want to save changes made to this recipe before creating a new recipe?" ) ), + i18n( "Unsaved changes" ) ) ) { + case KMessageBox::Yes: + inputPanel->save(); + break; + case KMessageBox::No: + break; + case KMessageBox::Cancel: + return ; + } + } + + inputPanel->newRecipe(); + slotSetPanel( RecipeEdit ); +} + +void KrecipesView::createNewElement( void ) +{ + //this is inconstant as the program stands... + /*if (rightPanel->visiblePanel())==4) //Properties Panel is the active one + { + propertiesPanel->createNewProperty(); + } + else*/{ + createNewRecipe(); + } +} + +void KrecipesView::wizard( bool force ) +{ + TDEConfig * config = kapp->config(); + config->setGroup( "Wizard" ); + bool setupDone = config->readBoolEntry( "SystemSetup", false ); + + TQString setupVersion = config->readEntry( "Version", "0.3" ); // By default assume it's 0.3. This parameter didn't exist in that version yet. + + if ( !setupDone || ( setupVersion.toDouble() < 0.5 ) || force ) // The config structure changed in version 0.4 to have DBType and Config Structure version + { + + bool setupUser, initData, doUSDAImport, adminEnabled; + TQString adminUser, adminPass, user, pass, host, client, dbName; + int port; + bool isRemote; + + SetupWizard *setupWizard = new SetupWizard( this ); + if ( setupWizard->exec() == TQDialog::Accepted ) + { + TDEConfig * config; + config = kapp->config(); + config->sync(); + config->setGroup( "DBType" ); + dbType = config->readEntry( "Type", "SQLite" ); + + kdDebug() << "Setting up" << endl; + setupWizard->getOptions( setupUser, initData, doUSDAImport ); + + // Setup user if necessary + if ( ( dbType == "MySQL" || dbType == "PostgreSQL" ) && setupUser ) // Don't setup user if checkbox of existing user... was set + { + kdDebug() << "Setting up user\n"; + setupWizard->getAdminInfo( adminEnabled, adminUser, adminPass, dbType ); + setupWizard->getServerInfo( isRemote, host, client, dbName, user, pass, port ); + + if ( !adminEnabled ) // Use root without password + { + kdDebug() << "Using default admin\n"; + if ( dbType == "MySQL" ) + adminUser = "root"; + else if ( dbType == "PostgreSQL" ) + adminUser = "postgres"; + adminPass = TQString::null; + } + if ( !isRemote ) // Use localhost + { + kdDebug() << "Using localhost\n"; + host = "localhost"; + client = "localhost"; + } + + setupUserPermissions( host, client, dbName, user, pass, adminUser, adminPass, port ); + } + + // Initialize database with data if requested + if ( initData ) { + setupWizard->getServerInfo( isRemote, host, client, dbName, user, pass, port ); + initializeData( host, dbName, user, pass, port ); // Populate data as normal user + } + + if ( doUSDAImport ) { + // Open the DB first + setupWizard->getServerInfo( isRemote, host, client, dbName, user, pass, port ); //only used if needed by backend + RecipeDB *db = RecipeDB::createDatabase( dbType, host, user, pass, dbName, port, dbName ); + + // Import the data + if ( db ) { + db->connect(); + + if ( db->ok() ) { + ProgressInterface pi(this); + pi.listenOn(db); + db->importUSDADatabase(); + } + + //close the database whether ok() or not + delete db; + } + } + + //we can do a faster usda import if this is done after it + if ( initData ) { + RecipeDB *db = RecipeDB::createDatabase( dbType, host, user, pass, dbName, port, dbName ); + if ( db ) { + db->connect(); + + if ( db->ok() ) { + db->importSamples(); + } + + //close the database whether ok() or not + delete db; + } + } + + } + delete setupWizard; + } +} + + +void KrecipesView::setupUserPermissions( const TQString &host, const TQString &client, const TQString &dbName, const TQString &newUser, const TQString &newPass, const TQString &adminUser, const TQString &adminPass, int port ) +{ + TQString user = adminUser; + TQString pass = adminPass; + if ( user.isNull() ) { + pass = TQString::null; + + if ( dbType == "PostgreSQL" ) + user = "postgres"; + else if ( dbType == "MySQL" ) + user = "root"; + + kdDebug() << "Open db as " << user << ", with no password\n"; + } + else + kdDebug() << "Open db as:" << user << ",*** with password ****\n"; + + RecipeDB *db = RecipeDB::createDatabase( dbType, host, user, pass, dbName, port, dbName ); + if ( db ) { + db->connect(true,false);//create the database, but no tables (do that when connected as the user) + if ( db->ok() ) + db->givePermissions( dbName, newUser, newPass, client ); // give permissions to the user + else + questionRerunWizard( db->err(), i18n( "Unable to setup database" ) ); + } + + delete db; //it closes the db automatically +} + + +void KrecipesView::initializeData( const TQString &host, const TQString &dbName, const TQString &user, const TQString &pass, int port ) +{ + RecipeDB * db = RecipeDB::createDatabase( dbType, host, user, pass, dbName, port, dbName ); + if ( !db ) { + kdError() << i18n( "Code error. No DB support has been included. Exiting" ) << endl; + kapp->exit( 1 ); + } + + db->connect(); + + if ( db->ok() ) { + db->emptyData(); + db->initializeData(); + } + + delete db; +} + +void KrecipesView::addRecipeButton( TQWidget *w, const TQString &title ) +{ + recipeWidget = w; + TDEIconLoader il; + if ( !recipeButton ) { + recipeButton = new KreMenuButton( leftPanel, RecipeEdit ); + + recipeButton->setIconSet( il.loadIconSet( "document-save", TDEIcon::Small ) ); + + TQString short_title = title.left( 20 ); + if ( title.length() > 20 ) + short_title.append( "..." ); + + recipeButton->setTitle( short_title ); + + buttonsList->append( recipeButton ); + leftPanel->highlightButton( recipeButton ); + + connect( recipeButton, TQ_SIGNAL( clicked() ), this, TQ_SLOT( switchToRecipe() ) ); + connect( ( RecipeInputDialog * ) w, TQ_SIGNAL( titleChanged( const TQString& ) ), recipeButton, TQ_SLOT( setTitle( const TQString& ) ) ); + } + +} + +void KrecipesView::switchToRecipe( void ) +{ + slotSetPanel( RecipeEdit ); +} + +void KrecipesView::closeRecipe( void ) +{ + slotSetPanel( SelectP ); + buttonsList->removeLast(); + recipeButton = 0; +} + +//Needed to make sure that the raise() is done after the construction of all the widgets, otherwise childEvent in the PanelDeco is called only _after_ the raise(), and can't be shown. + +void KrecipesView::show ( void ) +{ + slotSetPanel( SelectP ); + TQWidget::show(); +} + +void KrecipesView::showRecipe( int recipeID ) +{ + TQValueList<int> ids; + ids << recipeID; + showRecipes( ids ); +} + +void KrecipesView::showRecipes( const TQValueList<int> &recipeIDs ) +{ + if ( viewPanel->loadRecipes( recipeIDs ) ) + slotSetPanel( RecipeView ); +} + +void KrecipesView::activateContextHelp() +{ + switch ( m_activePanel ) { + case RecipeView: + //kapp->invokeHelp(""); + break; + + case SelectP: + kapp->invokeHelp("find-edit"); + break; + + case ShoppingP: + kapp->invokeHelp("shopping-list"); + break; + + case DietP: + kapp->invokeHelp("diet-helper"); + break; + + case MatcherP: + kapp->invokeHelp("ingredient-matcher"); + break; + + case RecipeEdit: + kapp->invokeHelp("enter-edit-recipes"); + break; + + case IngredientsP: + kapp->invokeHelp("ingredients-component"); + break; + + case PropertiesP: + kapp->invokeHelp("properties-component"); + break; + + case UnitsP: + kapp->invokeHelp("units-component"); + break; + + case PrepMethodsP: + kapp->invokeHelp("prep-methods"); + break; + + case CategoriesP: + kapp->invokeHelp("categories-component"); + break; + + case AuthorsP: + kapp->invokeHelp("authors-component"); + break; + } +} + +void KrecipesView::panelRaised( TQWidget *w, TQWidget *old_w ) +{ + emit panelShown( panelMap[ old_w ], false ); + emit panelShown( panelMap[ w ], true ); +} + + +void KrecipesView::createShoppingListFromDiet( void ) +{ + shoppingListPanel->createShopping( dietPanel->dietList() ); + slotSetPanel( ShoppingP ); +} + +void KrecipesView::moveTipButton( int, int ) +{ + contextButton->setGeometry( leftPanel->width() - 42, leftPanel->height() - 42, 32, 32 ); +} + +void KrecipesView::resizeRightPane( int lpw, int ) +{ + TQSize rpsize = rightPanel->size(); + TQPoint rpplace = rightPanel->pos(); + rpsize.setWidth( width() - lpw ); + rpplace.setX( lpw ); + rightPanel->move( rpplace ); + rightPanel->resize( rpsize ); + +} + + + +void KrecipesView::initDatabase( TDEConfig *config ) +{ + + // Read the database type + config->sync(); + config->setGroup( "DBType" ); + dbType = checkCorrectDBType( config ); + + + + // Open the database + database = RecipeDB::createDatabase( dbType ); + if ( !database ) { + // No DB type has been enabled(should not happen at all, but just in case) + + kdError() << i18n( "Code error. No DB support was built in. Exiting" ) << endl; + kapp->exit( 1 ); + } + + database->connect(); + + while ( !database->ok() ) { + // Ask the user if he wants to rerun the wizard + bool rerun = questionRerunWizard( database->err(), i18n( "Unable to open database" ) ); + if ( !rerun ) + break; + + // Reread the configuration file. + // The user may have changed the data and/or DB type + + config->sync(); + config->setGroup( "DBType" ); + dbType = checkCorrectDBType( config ); + + delete database; + database = RecipeDB::createDatabase( dbType ); + if ( database ) + database->connect(); + else { + // No DB type has been enabled (should not happen at all, but just in case) + + kdError() << i18n( "Code error. No DB support was built in. Exiting" ) << endl; + kapp->exit( 1 ); + break; + } + } + kdDebug() << i18n( "DB started correctly\n" ).latin1(); +} + +TQString KrecipesView::checkCorrectDBType( TDEConfig *config ) +{ + dbType = config->readEntry( "Type", "SQLite" ); + + while ( ( dbType != "SQLite" ) && ( dbType != "MySQL" ) && ( dbType != "PostgreSQL" ) ) { + questionRerunWizard( i18n( "The configured database type (%1) is unsupported." ).arg( dbType ), i18n( "Unsupported database type. Database must be either MySQL, SQLite, or PostgreSQL." ) ); + + // Read the database setup again + + config = kapp->config(); + config->sync(); + config->setGroup( "DBType" ); + dbType = config->readEntry( "Type", "SQLite" ); + } + return ( dbType ); +} + +void KrecipesView::reloadDisplay() +{ + viewPanel->reload(); +} + +void KrecipesView::editRecipe() +{ + KrePanel vis_panel = panelMap[ rightPanel->visiblePanel() ]; + + switch ( vis_panel ) { + case RecipeView: + actionRecipe( viewPanel->currentRecipes() [ 0 ], 1 ); + break; + case SelectP: + selectPanel->getActionsHandler()->edit(); + break; + default: + break; + } +} + +void KrecipesView::reload() +{ + viewPanel->reload(); + selectPanel->reload( ForceReload ); + shoppingListPanel->reload( ReloadIfPopulated ); + ingredientsPanel->reload( ReloadIfPopulated ); + propertiesPanel->reload(); + unitsPanel->reload( ReloadIfPopulated ); + dietPanel->reload( ReloadIfPopulated ); + authorsPanel->reload( ReloadIfPopulated ); + categoriesPanel->reload( ReloadIfPopulated ); + ingredientMatcherPanel->reload( ReloadIfPopulated ); + prepMethodsPanel->reload( ReloadIfPopulated ); +} + +DCOPRef KrecipesView::currentDatabase() const +{ + return DCOPRef(database); +} + + +#include "krecipesview.moc" diff --git a/src/krecipesview.h b/src/krecipesview.h new file mode 100644 index 0000000..01d22c0 --- /dev/null +++ b/src/krecipesview.h @@ -0,0 +1,215 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro <[email protected]> * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef KRECIPESVIEW_H +#define KRECIPESVIEW_H + +#include <tqbitmap.h> +#include <tqbuttongroup.h> +#include <tqevent.h> +#include <tqlabel.h> +#include <tqpushbutton.h> +#include <tqtooltip.h> +#include <tqhbox.h> +#include <tqvaluelist.h> +#include <tqvbox.h> +#include <tqwidget.h> +#include <tqwidgetstack.h> + +#include <kiconloader.h> +#include <kpixmap.h> +#include <kpixmapeffect.h> +#include <kimageeffect.h> +#include <tdeparts/part.h> +#include <tdestyle.h> +#include <ktextbrowser.h> + +#include "importers/baseimporter.h" +#include "backends/recipedb.h" +#include "krecipesiface.h" + +class AuthorsDialog; +class PrepMethodsDialog; +class CategoriesEditorDialog; +class DietWizardDialog; +class IngredientsDialog; +class Menu; +class KreMenu; +class KreMenuButton; +class IngredientMatcherDialog; +class PanelDeco; +class PropertiesDialog; +class TQPainter; +class RecipeInputDialog; +class RecipeViewDialog; +class SelectRecipeDialog; +class ShoppingListDialog; +class UnitsDialog; +typedef TQValueList <Menu>::Iterator MenuId; + + +/** + * This is the main view class for Krecipes. Most of the non-menu, + * non-toolbar, and non-statusbar (e.g., non frame) GUI code should go + * here. + * + * This krecipes uses an HTML component as an example. + * + * @short Main view + * @author Unai Garro <[email protected]> + * @version 0.4 + */ + + +// Declarations + + + +// Some constants +enum KrePanel {SelectP = 0, ShoppingP, DietP, MatcherP, IngredientsP, PropertiesP, UnitsP, PrepMethodsP, CategoriesP, AuthorsP, RecipeEdit, RecipeView }; + + +// Class KrecipesView +class KrecipesView : public TQVBox, virtual public KrecipesIface +{ + TQ_OBJECT +public: + /** + * Default constructor + */ + KrecipesView( TQWidget *parent ); + + /** + * Destructor + */ + virtual ~KrecipesView(); + + virtual DCOPRef currentDatabase() const; + RecipeDB *database; + + /** + * Print this view to any medium -- paper or not + */ + void print(); + + virtual void show ( void ); //Needed to make sure that the raise() is done after the construction of all the widgets, otherwise childEvent in the PanelDeco is called only _after_ the raise(), and can't be shown. + +signals: + /** + * Use this signal to change the content of the statusbar + */ + void signalChangeStatusbar( const TQString& text ); + + /** + * Use this signal to change the content of the caption + */ + void signalChangeCaption( const TQString& text ); + + void panelShown( KrePanel, bool ); + + +public: + + // public widgets + RecipeInputDialog *inputPanel; + RecipeViewDialog *viewPanel; + SelectRecipeDialog *selectPanel; + IngredientsDialog *ingredientsPanel; + PropertiesDialog *propertiesPanel; + UnitsDialog* unitsPanel; + ShoppingListDialog* shoppingListPanel; + DietWizardDialog* dietPanel; + CategoriesEditorDialog *categoriesPanel; + AuthorsDialog *authorsPanel; + PrepMethodsDialog *prepMethodsPanel; + IngredientMatcherDialog *ingredientMatcherPanel; + + // public methods + void createNewRecipe( void ); + void createNewElement( void ); + + void exportRecipes( const TQValueList<int> &ids ); + +private: + + // Internal methods + TQString checkCorrectDBType( TDEConfig *config ); + void initializeData( const TQString &host, const TQString &dbName, const TQString &user, const TQString &pass, int port ); + void initDatabase( TDEConfig *config ); + bool questionRerunWizard( const TQString &message, const TQString &error = "" ); + void setupUserPermissions( const TQString &host, const TQString &client, const TQString &dbName, const TQString &newUser, const TQString &newPass, const TQString &adminUser = TQString::null, const TQString &adminPass = TQString::null, int port = 0 ); + void wizard( bool force = false ); + + + + // Widgets + TQHBox *splitter; + KreMenu *leftPanel; + MenuId dataMenu; + PanelDeco *rightPanel; + TQPtrList<KreMenuButton> *buttonsList; + KreMenuButton *button0; + KreMenuButton *button1; + KreMenuButton *button2; + KreMenuButton *button3; + KreMenuButton *button4; + KreMenuButton *button5; + KreMenuButton *button6; + KreMenuButton *button7; + KreMenuButton *button8; + KreMenuButton *button9; + TQPushButton* contextButton; + + KreMenuButton *recipeButton; + TQWidget *recipeWidget; + + // Internal variables + TQString dbType; + KrePanel m_activePanel; + + TQMap<TQWidget*, KrePanel> panelMap; + + // i18n + void translate(); + + +signals: + void enableSaveOption( bool en ); + void recipeSelected( bool ); + +public slots: + bool save( void ); + void exportRecipe(); + void exportToClipboard(); + void reloadDisplay(); + virtual void reload(); + void activateContextHelp(); + +private slots: + void actionRecipe( int recipeID, int action ); + void actionRecipes( const TQValueList<int> &ids, int action ); + void addRecipeButton( TQWidget *w, const TQString &title ); + void closeRecipe( void ); + void showRecipe( int recipeID ); + void showRecipes( const TQValueList<int> &recipeIDs ); + void slotSetTitle( const TQString& title ); + void slotSetPanel( KrePanel ); + void switchToRecipe( void ); + void createShoppingListFromDiet( void ); + void moveTipButton( int, int ); + void resizeRightPane( int lpw, int lph ); + void panelRaised( TQWidget *w, TQWidget *old_w ); + void editRecipe(); +}; + + +#endif // KRECIPESVIEW_H diff --git a/src/krepagelayout.cpp b/src/krepagelayout.cpp new file mode 100644 index 0000000..72f4a6b --- /dev/null +++ b/src/krepagelayout.cpp @@ -0,0 +1,242 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999 Torben Weis <[email protected]> + Copyright 2002, 2003 David Faure <[email protected]> + Copyright 2003 Nicolas GOUTTE <[email protected]> + Copyright 2005 Jason Kivlighn <[email protected]> + + This library is free software; you can redistribute it and/or + modify it under the terms of the GNU Library General Public + License as published by the Free Software Foundation; either + version 2 of the License, or (at your option) any later version. + + This library 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 Library General Public License + along with this library; see the file COPYING.LIB. If not, write to + the Free Software Foundation, Inc., 59 Temple Place - Suite 330, + Boston, MA 02111-1307, USA. +*/ +#include "krepagelayout.h" + +#include <tdelocale.h> +#include <kprinter.h> +#include <kdebug.h> +#include <tdeglobal.h> + +#include <tqdom.h> + +#include "kreunit.h" + +// paper formats ( mm ) +#define PG_A3_WIDTH 297.0 +#define PG_A3_HEIGHT 420.0 +#define PG_A4_WIDTH 210.0 +#define PG_A4_HEIGHT 297.0 +#define PG_A5_WIDTH 148.0 +#define PG_A5_HEIGHT 210.0 +#define PG_B5_WIDTH 182.0 +#define PG_B5_HEIGHT 257.0 +#define PG_US_LETTER_WIDTH 216.0 +#define PG_US_LETTER_HEIGHT 279.0 +#define PG_US_LEGAL_WIDTH 216.0 +#define PG_US_LEGAL_HEIGHT 356.0 +#define PG_US_EXECUTIVE_WIDTH 191.0 +#define PG_US_EXECUTIVE_HEIGHT 254.0 + +TQDomElement KoPageLayout::saveKreFormat( TQDomDocument &doc ) const +{ + TQDomElement style = doc.createElement( "page-layout-properties" ); + style.setAttribute("page-width", ptWidth); + style.setAttribute("page-height", ptHeight); + style.setAttribute("margin-left", ptLeft); + style.setAttribute("margin-right", ptRight); + style.setAttribute("margin-top", ptTop); + style.setAttribute("margin-bottom", ptBottom); + style.setAttribute("print-orientation", (orientation == PG_LANDSCAPE ? "landscape" : "portrait"));kdDebug()<<"margin-left: "<<ptLeft<<endl; + return style; +} + +void KoPageLayout::loadKreFormat(const TQDomElement &style) +{ + if ( !style.isNull() ) + { + ptWidth = KoUnit::parseValue(style.attribute("page-width", TQString::null ) ); + ptHeight = KoUnit::parseValue(style.attribute( "page-height", TQString::null ) ); + if (style.attribute( "print-orientation", TQString::null)=="portrait") + orientation=PG_PORTRAIT; + else + orientation=PG_LANDSCAPE; + ptRight = KoUnit::parseValue( style.attribute( "margin-right", TQString::null ) ); + ptBottom = KoUnit::parseValue( style.attribute( "margin-bottom", TQString::null ) ); + ptLeft = KoUnit::parseValue( style.attribute( "margin-left", TQString::null ) ); + ptTop = KoUnit::parseValue( style.attribute( "margin-top", TQString::null ) ); + // guessFormat takes millimeters + if ( orientation == PG_LANDSCAPE ) + format = KoPageFormat::guessFormat( POINT_TO_MM(ptHeight), POINT_TO_MM(ptWidth) ); + else + format = KoPageFormat::guessFormat( POINT_TO_MM(ptWidth), POINT_TO_MM(ptHeight) ); + } +} + + +KoPageLayout KoPageLayout::standardLayout() +{ + KoPageLayout layout; + layout.format = KoPageFormat::defaultFormat(); + layout.orientation = PG_PORTRAIT; + layout.ptWidth = MM_TO_POINT( KoPageFormat::width( layout.format, layout.orientation ) ); + layout.ptHeight = MM_TO_POINT( KoPageFormat::height( layout.format, layout.orientation ) ); + layout.ptLeft = MM_TO_POINT( 20.0 ); + layout.ptRight = MM_TO_POINT( 20.0 ); + layout.ptTop = MM_TO_POINT( 20.0 ); + layout.ptBottom = MM_TO_POINT( 20.0 ); + return layout; +} + +struct PageFormatInfo +{ + KoFormat format; + KPrinter::PageSize kprinter; + const char* shortName; // Short name + const char* descriptiveName; // Full name, which will be translated + double width; // in mm + double height; // in mm +}; + +// NOTES: +// - the width and height of non-ISO formats are rounded +// http://en.wikipedia.org/wiki/Paper_size can help +// - the comments "should be..." indicates the exact values if the inch sizes would be multiplied by 25.4 mm/inch + +const PageFormatInfo pageFormatInfo[]= +{ + { PG_DIN_A3, KPrinter::A3, "A3", I18N_NOOP("ISO A3"), 297.0, 420.0 }, + { PG_DIN_A4, KPrinter::A4, "A4", I18N_NOOP("ISO A4"), 210.0, 297.0 }, + { PG_DIN_A5, KPrinter::A5, "A5", I18N_NOOP("ISO A5"), 148.0, 210.0 }, + { PG_US_LETTER, KPrinter::Letter, "Letter", I18N_NOOP("US Letter"), 215.9, 279.4 }, + { PG_US_LEGAL, KPrinter::Legal, "Legal", I18N_NOOP("US Legal"), 215.9, 355.6 }, + { PG_SCREEN, KPrinter::A4, "Screen", I18N_NOOP("Screen"), PG_A4_HEIGHT, PG_A4_WIDTH }, // Custom, so fall back to A4 + { PG_CUSTOM, KPrinter::A4, "Custom", I18N_NOOP("Custom"), PG_A4_WIDTH, PG_A4_HEIGHT }, // Custom, so fall back to A4 + { PG_DIN_B5, KPrinter::B5, "B5", I18N_NOOP("ISO B5"), 182.0, 257.0 }, + // Hmm, wikipedia says 184.15 * 266.7 for executive ! + { PG_US_EXECUTIVE, KPrinter::Executive, "Executive", I18N_NOOP("US Executive"), 191.0, 254.0 }, // should be 190.5 mm x 254.0 mm + { PG_DIN_A0, KPrinter::A0, "A0", I18N_NOOP("ISO A0"), 841.0, 1189.0 }, + { PG_DIN_A1, KPrinter::A1, "A1", I18N_NOOP("ISO A1"), 594.0, 841.0 }, + { PG_DIN_A2, KPrinter::A2, "A2", I18N_NOOP("ISO A2"), 420.0, 594.0 }, + { PG_DIN_A6, KPrinter::A6, "A6", I18N_NOOP("ISO A6"), 105.0, 148.0 }, + { PG_DIN_A7, KPrinter::A7, "A7", I18N_NOOP("ISO A7"), 74.0, 105.0 }, + { PG_DIN_A8, KPrinter::A8, "A8", I18N_NOOP("ISO A8"), 52.0, 74.0 }, + { PG_DIN_A9, KPrinter::A9, "A9", I18N_NOOP("ISO A9"), 37.0, 52.0 }, + { PG_DIN_B0, KPrinter::B0, "B0", I18N_NOOP("ISO B0"), 1030.0, 1456.0 }, + { PG_DIN_B1, KPrinter::B1, "B1", I18N_NOOP("ISO B1"), 728.0, 1030.0 }, + { PG_DIN_B10, KPrinter::B10, "B10", I18N_NOOP("ISO B10"), 32.0, 45.0 }, + { PG_DIN_B2, KPrinter::B2, "B2", I18N_NOOP("ISO B2"), 515.0, 728.0 }, + { PG_DIN_B3, KPrinter::B3, "B3", I18N_NOOP("ISO B3"), 364.0, 515.0 }, + { PG_DIN_B4, KPrinter::B4, "B4", I18N_NOOP("ISO B4"), 257.0, 364.0 }, + { PG_DIN_B6, KPrinter::B6, "B6", I18N_NOOP("ISO B6"), 128.0, 182.0 }, + { PG_ISO_C5, KPrinter::C5E, "C5", I18N_NOOP("ISO C5"), 163.0, 229.0 }, // Some sources tells: 162 mm x 228 mm + { PG_US_COMM10, KPrinter::Comm10E, "Comm10", I18N_NOOP("US Common 10"), 105.0, 241.0 }, // should be 104.775 mm x 241.3 mm + { PG_ISO_DL, KPrinter::DLE, "DL", I18N_NOOP("ISO DL"), 110.0, 220.0 }, + { PG_US_FOLIO, KPrinter::Folio, "Folio", I18N_NOOP("US Folio"), 210.0, 330.0 }, // should be 209.54 mm x 330.2 mm + { PG_US_LEDGER, KPrinter::Ledger, "Ledger", I18N_NOOP("US Ledger"), 432.0, 279.0 }, // should be 431.8 mm x 297.4 mm + { PG_US_TABLOID, KPrinter::Tabloid, "Tabloid", I18N_NOOP("US Tabloid"), 279.0, 432.0 } // should be 297.4 mm x 431.8 mm +}; + +int KoPageFormat::printerPageSize( KoFormat format ) +{ + if ( format == PG_SCREEN ) + { + kdWarning() << "You use the page layout SCREEN. Printing in DIN A4 LANDSCAPE." << endl; + return KPrinter::A4; + } + else if ( format == PG_CUSTOM ) + { + kdWarning() << "The used page layout (CUSTOM) is not supported by KPrinter. Printing in A4." << endl; + return KPrinter::A4; + } + else if ( format <= PG_LAST_FORMAT ) + return pageFormatInfo[ format ].kprinter; + else + return KPrinter::A4; +} + +double KoPageFormat::width( KoFormat format, KoOrientation orientation ) +{ + if ( orientation == PG_LANDSCAPE ) + return height( format, PG_PORTRAIT ); + if ( format <= PG_LAST_FORMAT ) + return pageFormatInfo[ format ].width; + return PG_A4_WIDTH; // should never happen +} + +double KoPageFormat::height( KoFormat format, KoOrientation orientation ) +{ + if ( orientation == PG_LANDSCAPE ) + return width( format, PG_PORTRAIT ); + if ( format <= PG_LAST_FORMAT ) + return pageFormatInfo[ format ].height; + return PG_A4_HEIGHT; +} + +KoFormat KoPageFormat::guessFormat( double width, double height ) +{ + for ( int i = 0 ; i <= PG_LAST_FORMAT ; ++i ) + { + // We have some tolerance. 1pt is a third of a mm, this is + // barely noticeable for a page size. + if ( i != PG_CUSTOM + && kAbs( width - pageFormatInfo[i].width ) < 1.0 + && kAbs( height - pageFormatInfo[i].height ) < 1.0 ) + return static_cast<KoFormat>(i); + } + return PG_CUSTOM; +} + +TQString KoPageFormat::formatString( KoFormat format ) +{ + if ( format <= PG_LAST_FORMAT ) + return TQString::fromLatin1( pageFormatInfo[ format ].shortName ); + return TQString::fromLatin1( "A4" ); +} + +KoFormat KoPageFormat::formatFromString( const TQString & string ) +{ + for ( int i = 0 ; i <= PG_LAST_FORMAT ; ++i ) + { + if (string == TQString::fromLatin1( pageFormatInfo[ i ].shortName )) + return pageFormatInfo[ i ].format; + } + // We do not know the format name, so we have a custom format + return PG_CUSTOM; +} + +KoFormat KoPageFormat::defaultFormat() +{ + int kprinter = TDEGlobal::locale()->pageSize(); + for ( int i = 0 ; i <= PG_LAST_FORMAT ; ++i ) + { + if ( pageFormatInfo[ i ].kprinter == kprinter ) + return static_cast<KoFormat>(i); + } + return PG_DIN_A4; +} + +TQString KoPageFormat::name( KoFormat format ) +{ + if ( format <= PG_LAST_FORMAT ) + return i18n( pageFormatInfo[ format ].descriptiveName ); + return i18n( pageFormatInfo[ PG_DIN_A4 ].descriptiveName ); +} + +TQStringList KoPageFormat::allFormats() +{ + TQStringList lst; + for ( int i = 0 ; i <= PG_LAST_FORMAT ; ++i ) + { + lst << i18n( pageFormatInfo[ i ].descriptiveName ); + } + return lst; +} diff --git a/src/krepagelayout.h b/src/krepagelayout.h new file mode 100644 index 0000000..2d788f2 --- /dev/null +++ b/src/krepagelayout.h @@ -0,0 +1,257 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999 Torben Weis <[email protected]> + Copyright 2002, 2003 David Faure <[email protected]> + Copyright 2003 Nicolas GOUTTE <[email protected]> + Copyright 2005 Jason Kivlighn <[email protected]> + + This library is free software; you can redistribute it and/or + modify it under the terms of the GNU Library General Public + License as published by the Free Software Foundation; either + version 2 of the License, or (at your option) any later version. + + This library 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 Library General Public License + along with this library; see the file COPYING.LIB. If not, write to + the Free Software Foundation, Inc., 59 Temple Place - Suite 330, + Boston, MA 02111-1307, USA. +*/ + +#ifndef KREPAGELAYOUT_H +#define KREPAGELAYOUT_H + +#include <tqstringlist.h> + +class TQDomElement; +class TQDomDocument; + +/** + * @brief Represents the paper format a document shall be printed on. + * + * For compatibility reasons, and because of screen and custom, + * this enum doesn't map to TQPrinter::PageSize but KoPageFormat::printerPageSize + * does the conversion. + * + * @todo convert DIN to ISO in the names + */ +enum KoFormat { + PG_DIN_A3 = 0, + PG_DIN_A4 = 1, + PG_DIN_A5 = 2, + PG_US_LETTER = 3, + PG_US_LEGAL = 4, + PG_SCREEN = 5, + PG_CUSTOM = 6, + PG_DIN_B5 = 7, + PG_US_EXECUTIVE = 8, + PG_DIN_A0 = 9, + PG_DIN_A1 = 10, + PG_DIN_A2 = 11, + PG_DIN_A6 = 12, + PG_DIN_A7 = 13, + PG_DIN_A8 = 14, + PG_DIN_A9 = 15, + PG_DIN_B0 = 16, + PG_DIN_B1 = 17, + PG_DIN_B10 = 18, + PG_DIN_B2 = 19, + PG_DIN_B3 = 20, + PG_DIN_B4 = 21, + PG_DIN_B6 = 22, + PG_ISO_C5 = 23, + PG_US_COMM10 = 24, + PG_ISO_DL = 25, + PG_US_FOLIO = 26, + PG_US_LEDGER = 27, + PG_US_TABLOID = 28, + // update the number below and the static arrays if you add more values to the enum + PG_LAST_FORMAT = PG_US_TABLOID // used by koPageLayout.cpp +}; + +/** + * Represents the orientation of a printed document. + */ +enum KoOrientation { + PG_PORTRAIT = 0, + PG_LANDSCAPE = 1 +}; + +namespace KoPageFormat +{ + /** + * @brief Convert a KoFormat into a KPrinter::PageSize. + * + * If format is 'screen' it will use A4 landscape. + * If format is 'custom' it will use A4 portrait. + * (you may want to take care of those cases separately). + * Usually passed to KPrinter::setPageSize(). + * + * @note We return int instead of the enum to avoid including kprinter.h + */ + int /*KPrinter::PageSize*/ printerPageSize( KoFormat format ); + + /** + * Returns the width (in mm) for a given page format and orientation + * 'Custom' isn't supported by this function, obviously. + */ + double width( KoFormat format, KoOrientation orientation ); + + /** + * Returns the height (in mm) for a given page format and orientation + * 'Custom' isn't supported by this function, obviously. + */ + double height( KoFormat format, KoOrientation orientation ); + + /** + * Returns the internal name of the given page format. + * Use for saving. + */ + TQString formatString( KoFormat format ); + + /** + * Convert a format string (internal name) to a page format value. + * Use for loading. + */ + KoFormat formatFromString( const TQString & string ); + + /** + * Returns the default format (based on the KControl settings) + */ + KoFormat defaultFormat(); + + /** + * Returns the translated name of the given page format. + * Use for showing the user. + */ + TQString name( KoFormat format ); + + /** + * Lists the translated names of all the available formats + */ + TQStringList allFormats(); + + /** + * Try to find the paper format for the given width and height (in mm). + * Useful to some import filters. + */ + KoFormat guessFormat( double width, double height ); +} + + +/** + * @brief Header/Footer type. + * + * @note Yes, this should have been a bitfield, but there was only 0, 2, 3 in koffice-1.0. Don't ask why. + * In the long run this should be replaced with a more flexible repetition/section concept. + */ +enum KoHFType { + HF_SAME = 0, ///< 0: Header/Footer is the same on all pages + HF_FIRST_EO_DIFF = 1, ///< 1: Header/Footer is different on first, even and odd pages (2&3) + HF_FIRST_DIFF = 2, ///< 2: Header/Footer for the first page differs + HF_EO_DIFF = 3 ///< 3: Header/Footer for even - odd pages are different +}; + +/** + * This structure defines the page layout, including + * its size in pt, its format (e.g. A4), orientation, unit, margins etc. + */ +struct KoPageLayout +{ + /** Page format */ + KoFormat format; + /** Page orientation */ + KoOrientation orientation; + + /** Page width in pt */ + double ptWidth; + /** Page height in pt */ + double ptHeight; + /** Left margin in pt */ + double ptLeft; + /** Right margin in pt */ + double ptRight; + /** Top margin in pt */ + double ptTop; + /** Bottom margin in pt */ + double ptBottom; + + bool operator==( const KoPageLayout& l ) const { + return ( ptWidth == l.ptWidth && + ptHeight == l.ptHeight && + ptLeft == l.ptLeft && + ptRight == l.ptRight && + ptTop == l.ptTop && + ptBottom == l.ptBottom ); + } + bool operator!=( const KoPageLayout& l ) const { + return !( (*this) == l ); + } + + /** + * Save this page layout to the Krecipes layout format + */ + TQDomElement saveKreFormat( TQDomDocument &doc ) const; + + /** + * Load this page layout from the Krecipes layout format + */ + void loadKreFormat(const TQDomElement &style); + + /** + * @return a page layout with the default page size depending on the locale settings, + * default margins (2 cm), and portrait orientation. + * @since 1.4 + */ + static KoPageLayout standardLayout(); +}; + +/** structure for header-footer */ +struct KoHeadFoot +{ + TQString headLeft; + TQString headMid; + TQString headRight; + TQString footLeft; + TQString footMid; + TQString footRight; +}; + +/** structure for columns */ +struct KoColumns +{ + int columns; + double ptColumnSpacing; + bool operator==( const KoColumns& rhs ) const { + return columns == rhs.columns && + TQABS(ptColumnSpacing - rhs.ptColumnSpacing) <= 1E-10; + } + bool operator!=( const KoColumns& rhs ) const { + return columns != rhs.columns || + TQABS(ptColumnSpacing - rhs.ptColumnSpacing) > 1E-10; + } +}; + +/** structure for KWord header-footer */ +struct KoKWHeaderFooter +{ + KoHFType header; + KoHFType footer; + double ptHeaderBodySpacing; + double ptFooterBodySpacing; + double ptFootNoteBodySpacing; + bool operator==( const KoKWHeaderFooter& rhs ) const { + return header == rhs.header && footer == rhs.footer && + TQABS(ptHeaderBodySpacing - rhs.ptHeaderBodySpacing) <= 1E-10 && + TQABS(ptFooterBodySpacing - rhs.ptFooterBodySpacing) <= 1E-10 && + TQABS(ptFootNoteBodySpacing - rhs.ptFootNoteBodySpacing) <= 1E-10; + } + bool operator!=( const KoKWHeaderFooter& rhs ) const { + return !( *this == rhs ); + } +}; + +#endif /* KREPAGELAYOUT_H */ + diff --git a/src/kstartuplogo.cpp b/src/kstartuplogo.cpp new file mode 100644 index 0000000..350d705 --- /dev/null +++ b/src/kstartuplogo.cpp @@ -0,0 +1,69 @@ +/*************************************************************************** +* Copyright (C) 2003 * +* * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +* * +* Based on kstartuplogo from Umbrello http://uml.sourceforge.net * +***************************************************************************/ + +#include "kstartuplogo.h" + +#include <tdeconfig.h> +#include <tdeglobal.h> + +#include <tqcursor.h> + +TDEStartupLogo::TDEStartupLogo( TQWidget * parent, const char *name ) : TQWidget( parent, name, WStyle_NoBorder | WStyle_Customize | WDestructiveClose ), m_bReadyToHide( false ) +{ + TQString dataDir = locate( "data", "krecipes/pics/startlogo.png" ); + TQPixmap pm( dataDir ); + setBackgroundPixmap( pm ); + + resize(pm.size()); + TQRect desk = splashScreenDesktopGeometry(); + setGeometry( ( desk.width() / 2 ) - ( width() / 2 ) + desk.left(), + ( desk.height() / 2 ) - ( height() / 2 ) + desk.top(), + width(), height() ); +} + +TDEStartupLogo::~TDEStartupLogo() +{} + +void TDEStartupLogo::mousePressEvent( TQMouseEvent* ) +{ + if ( m_bReadyToHide ) { + hide(); + } +} + +/* This function is based on TDE's TDEGlobalSettings::splashScreenDesktopGeometry(). */ + +TQRect TDEStartupLogo::splashScreenDesktopGeometry() const +{ + TQDesktopWidget *dw = TQApplication::desktop(); + + if (dw->isVirtualDesktop()) { + TDEConfigGroup group(TDEGlobal::config(), "Windows"); + int scr = group.readNumEntry("Unmanaged", -3); + if (group.readBoolEntry("XineramaEnabled", true) && scr != -2) { + if (scr == -3) + scr = dw->screenNumber(TQCursor::pos()); + return dw->screenGeometry(scr); + } + else { + return dw->geometry(); + } + } + else { + return dw->geometry(); + } +} + +#include "kstartuplogo.moc" diff --git a/src/kstartuplogo.h b/src/kstartuplogo.h new file mode 100644 index 0000000..c4472cc --- /dev/null +++ b/src/kstartuplogo.h @@ -0,0 +1,49 @@ +/*************************************************************************** +* Copyright (C) 2003 * +* * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +* * +* Based on kstartuplogo from Umbrello http://uml.sourceforge.net * +***************************************************************************/ + + +#ifndef KSTARTUPLOGO_H +#define KSTARTUPLOGO_H + +#include <tqwidget.h> + +#include <tdeapplication.h> +#include <kstandarddirs.h> + +/** + * Displays a startup splash screen + */ +class TDEStartupLogo : public TQWidget +{ + TQ_OBJECT +public: + TDEStartupLogo( TQWidget *parent = 0, const char *name = 0 ); + ~TDEStartupLogo(); + void setHideEnabled( bool bEnabled ) + { + m_bReadyToHide = bEnabled; + }; +protected: + virtual void mousePressEvent( TQMouseEvent* ); + TQRect splashScreenDesktopGeometry() const; + bool m_bReadyToHide; +}; + +#endif + + + + + diff --git a/src/main.cpp b/src/main.cpp new file mode 100644 index 0000000..de93bf2 --- /dev/null +++ b/src/main.cpp @@ -0,0 +1,81 @@ + +/*************************************************************************** +* Copyright (C) 2003-2005 by * +* Unai Garro ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "krecipes.h" + +#include <iostream> + +#include <kuniqueapplication.h> +#include <tdeaboutdata.h> +#include <tdecmdlineargs.h> +#include <tdelocale.h> + +#include "convert_sqlite3.h" + +static const char *description = + I18N_NOOP( "The TDE Cookbook" ); + +static const char *version = "1.0-beta1"; + +static TDECmdLineOptions options[] = + { + { "convert-sqlite3", I18N_NOOP("Convert the current SQLite 2.x database to SQLite 3 and exit") , 0 }, + { 0, 0, 0 } + }; + +int main( int argc, char **argv ) +{ + TDEAboutData about( "krecipes", I18N_NOOP( "Krecipes" ), version, description, + TDEAboutData::License_GPL, I18N_NOOP( "(C) 2003 Unai Garro\n(C) 2004-2006 Jason Kivlighn\n\n___________\n\n\nThis product is RecipeML compatible.\n You can get more information about this file format in:\n http://www.formatdata.com/recipeml" ), 0, 0, "[email protected]" ); + about.addAuthor( "Unai Garro", 0, "[email protected]" ); + about.addAuthor( "Jason Kivlighn", 0, "[email protected]" ); + about.addAuthor( "Cyril Bosselut", 0, "[email protected]" ); + + about.addCredit( "Colleen Beamer", I18N_NOOP("Testing, bug reports, suggestions"), "[email protected]" ); + + about.setTranslator( I18N_NOOP( "INSERT YOUR NAME HERE" ), I18N_NOOP( "INSERT YOUR EMAIL ADDRESS" ) ); + TDECmdLineArgs::init( argc, argv, &about ); + TDECmdLineArgs::addCmdLineOptions( options ); + KUniqueApplication::addCmdLineOptions(); + + if ( !KUniqueApplication::start() ) { + std::cout << "Krecipes is already running!" << std::endl; + return 0; + } + + KUniqueApplication app; + + // see if we are starting with session management + if ( app.isRestored() ) { + RESTORE( Krecipes ); + } + else { + // no session.. just start up normally + TDECmdLineArgs *args = TDECmdLineArgs::parsedArgs(); + + TQApplication::flushX(); + + if ( args->isSet("convert-sqlite3") ) { + ConvertSQLite3(); + return 0; + } + + Krecipes * widget = new Krecipes; + app.setMainWidget( widget ); + widget->show(); + + args->clear(); + } + + return app.exec(); +} + diff --git a/src/mmdata.h b/src/mmdata.h new file mode 100644 index 0000000..c8ee398 --- /dev/null +++ b/src/mmdata.h @@ -0,0 +1,65 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef MMDATA_H +#define MMDATA_H + +/** This file contains each of the Meal-Master unit abbreviations, + * and their cooresponding expansions. + */ + +struct expand_unit_info +{ + const char *short_form; + const char *expanded_form; + const char *plural_expanded_form; +}; + +static expand_unit_info unit_info[] = { + {"bn", I18N_NOOP( "bunch" ), I18N_NOOP( "bunches" ) }, + {"c" , I18N_NOOP( "cup" ), I18N_NOOP( "cups" ) }, + {"cc", I18N_NOOP( "cubic cm" ), I18N_NOOP( "cubic cm" ) }, + {"cg", I18N_NOOP( "centigram" ), I18N_NOOP( "centigrams" ) }, + {"cl", I18N_NOOP( "centiliter" ), I18N_NOOP( "centiliters" ) }, + {"cn", I18N_NOOP( "can" ), I18N_NOOP( "cans" ) }, + {"ct", I18N_NOOP( "carton" ), I18N_NOOP( "cartons" ) }, + {"dg", I18N_NOOP( "decigram" ), I18N_NOOP( "decigrams" ) }, + {"dl", I18N_NOOP( "deciliter" ), I18N_NOOP( "deciliters" ) }, + {"dr", I18N_NOOP( "drop" ), I18N_NOOP( "drops" ) }, + {"ds", I18N_NOOP( "dash" ), I18N_NOOP( "dashes" ) }, + {"ea", I18N_NOOP( "each" ), I18N_NOOP( "each" ) }, + {"kg", I18N_NOOP( "kilogram" ), I18N_NOOP( "kilograms" ) }, + {"fl", I18N_NOOP( "fluid ounce" ), I18N_NOOP( "fluid ounces" ) }, + {"g" , I18N_NOOP( "gram" ), I18N_NOOP( "grams" ) }, + {"ga", I18N_NOOP( "gallon" ), I18N_NOOP( "gallons" ) }, + {"l" , I18N_NOOP( "liter" ), I18N_NOOP( "liters" ) }, + {"lb", I18N_NOOP( "pound" ), I18N_NOOP( "pounds" ) }, + {"lg", I18N_NOOP( "large" ), I18N_NOOP( "large" ) }, + {"md", I18N_NOOP( "medium" ), I18N_NOOP( "medium" ) }, + {"mg", I18N_NOOP( "milligram" ), I18N_NOOP( "milligrams" ) }, + {"ml", I18N_NOOP( "milliliter" ), I18N_NOOP( "milliliters" ) }, + {"pg", I18N_NOOP( "package" ), I18N_NOOP( "packages" ) }, + {"pk", I18N_NOOP( "package" ), I18N_NOOP( "packages" ) }, + {"pn", I18N_NOOP( "pinch" ), I18N_NOOP( "pinches" ) }, + {"pt", I18N_NOOP( "pint" ), I18N_NOOP( "pints" ) }, + {"oz", I18N_NOOP( "ounce" ), I18N_NOOP( "ounces" ) }, + {"qt", I18N_NOOP( "quart" ), I18N_NOOP( "quarts" ) }, + {"sl", I18N_NOOP( "slice" ), I18N_NOOP( "slices" ) }, + {"sm", I18N_NOOP( "small" ), I18N_NOOP( "small" ) }, + {"t" , I18N_NOOP( "teaspoon" ), I18N_NOOP( "teaspoons" ) }, + {"tb", I18N_NOOP( "tablespoon" ), I18N_NOOP( "tablespoons" ) }, + {"ts", I18N_NOOP( "teaspoon" ), I18N_NOOP( "teaspoons" ) }, + {"T" , I18N_NOOP( "tablespoon" ), I18N_NOOP( "tablespoons" ) }, + {"x" , I18N_NOOP( "per serving" ), I18N_NOOP( "per serving" ) }, + {"", "", ""}, + { 0, 0, 0 } + }; + +#endif //MMDATA_H diff --git a/src/pref.cpp b/src/pref.cpp new file mode 100644 index 0000000..c2575cf --- /dev/null +++ b/src/pref.cpp @@ -0,0 +1,678 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* Copyright (C) 2004-2006 Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "pref.h" + +#ifdef HAVE_CONFIG_H +#include "config.h" +#endif + +#include <tqlayout.h> +#include <tqlabel.h> +#include <tqhbox.h> +#include <tqbuttongroup.h> +#include <tqcheckbox.h> +#include <tqradiobutton.h> +#include <tqpushbutton.h> +#include <tqtooltip.h> +#include <tqwhatsthis.h> +#include <tqframe.h> +#include <tqcombobox.h> + +#include <tdeapplication.h> +#include <tdeconfig.h> +#include <kiconloader.h> +#include <tdelocale.h> +#include <kstandarddirs.h> +#include <tdefiledialog.h> +#include <knuminput.h> +#include <klineedit.h> +#include <kurlrequester.h> +#include <kdebug.h> + +KrecipesPreferences::KrecipesPreferences( TQWidget *parent ) + : KDialogBase( IconList, i18n( "Krecipes Preferences" ), + Help | Ok | Cancel, Ok, parent ) +{ + // this is the base class for your preferences dialog. it is now + // a TreeList dialog.. but there are a number of other + // possibilities (including Tab, Swallow, and just Plain) + TQFrame * frame; + + TDEConfig *config = TDEGlobal::config(); + config->setGroup( "DBType" ); + + TDEIconLoader il; + frame = addPage( i18n( "Server Settings" ), + TQString( i18n( "Database Server Options (%1)" ) ).arg( config->readEntry( "Type" ) ), + il.loadIcon( "network_local", TDEIcon::NoGroup, 32 ) ); + TQHBoxLayout* layout = new TQHBoxLayout( frame ); + m_pageServer = new ServerPrefs( frame ); + layout->addWidget( m_pageServer ); + m_helpMap.insert(0,"configure-server-settings"); + + frame = addPage( i18n( "Formatting" ), i18n( "Customize Formatting" ), il.loadIcon( "math_frac", TDEIcon::NoGroup, 32 ) ); + TQHBoxLayout* formatting_layout = new TQHBoxLayout( frame ); + m_pageNumbers = new NumbersPrefs( frame ); + formatting_layout->addWidget( m_pageNumbers ); + m_helpMap.insert(1,"custom-formatting"); + + frame = addPage( i18n( "Import/Export" ), i18n( "Recipe Import and Export Options" ), il.loadIcon( "go-down", TDEIcon::NoGroup, 32 ) ); + TQHBoxLayout* import_layout = new TQHBoxLayout( frame ); + m_pageImport = new ImportPrefs( frame ); + import_layout->addWidget( m_pageImport ); + m_helpMap.insert(2,"import-export-preference"); + + frame = addPage( i18n( "Performance" ), i18n( "Performance Options" ), il.loadIcon( "launch", TDEIcon::NoGroup, 32 ) ); + TQHBoxLayout* performance_layout = new TQHBoxLayout( frame ); + m_pagePerformance = new PerformancePrefs( frame ); + performance_layout->addWidget( m_pagePerformance ); + m_helpMap.insert(3,"configure-performance"); + + // Signals & Slots + connect ( this, TQ_SIGNAL( okClicked() ), this, TQ_SLOT( saveSettings() ) ); + +} + +void KrecipesPreferences::slotHelp() +{ + kapp->invokeHelp( m_helpMap[activePageIndex()] ); +} + + +MySQLServerPrefs::MySQLServerPrefs( TQWidget *parent ) : TQWidget( parent ) +{ + TQGridLayout * layout = new TQGridLayout( this, 1, 1, 0, 0 ); + layout->setSpacing( KDialog::spacingHint() ); + layout->setMargin( 0 ); + + TQSpacerItem* spacerTop = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( spacerTop, 0, 1 ); + TQSpacerItem* spacerLeft = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::Minimum ); + layout->addItem( spacerLeft, 1, 0 ); + + TQLabel* serverText = new TQLabel( i18n( "Server:" ), this ); + serverText->setFixedSize( TQSize( 100, 20 ) ); + serverText->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + layout->addWidget( serverText, 1, 1 ); + + serverEdit = new KLineEdit( this ); + serverEdit->setFixedSize( TQSize( 120, 20 ) ); + serverEdit->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + layout->addWidget( serverEdit, 1, 2 ); + + TQSpacerItem* spacerRow1 = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( spacerRow1, 2, 1 ); + + TQLabel* usernameText = new TQLabel( i18n( "Username:" ), this ); + usernameText->setFixedSize( TQSize( 100, 20 ) ); + usernameText->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + layout->addWidget( usernameText, 3, 1 ); + + usernameEdit = new KLineEdit( this ); + usernameEdit->setFixedSize( TQSize( 120, 20 ) ); + usernameEdit->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + layout->addWidget( usernameEdit, 3, 2 ); + + TQSpacerItem* spacerRow2 = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( spacerRow2, 4, 1 ); + + TQLabel* passwordText = new TQLabel( i18n( "Password:" ), this ); + passwordText->setFixedSize( TQSize( 100, 20 ) ); + passwordText->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + layout->addWidget( passwordText, 5, 1 ); + + passwordEdit = new KLineEdit( this ); + passwordEdit->setFixedSize( TQSize( 120, 20 ) ); + passwordEdit->setEchoMode( TQLineEdit::Password ); + passwordEdit->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + layout->addWidget( passwordEdit, 5, 2 ); + + TQSpacerItem* spacerRow3 = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( spacerRow3, 6, 1 ); + + TQLabel* portText = new TQLabel( i18n( "Port:" ), this ); + portText->setFixedSize( TQSize( 100, 20 ) ); + portText->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + layout->addWidget( portText, 7, 1 ); + + portEdit = new KIntNumInput( this ); + portEdit->setMinValue(0); + portEdit->setSpecialValueText( i18n("Default") ); + portEdit->setFixedSize( TQSize( 120, 20 ) ); + portEdit->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + layout->addWidget( portEdit, 7, 2 ); + + TQSpacerItem* spacerRow4 = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( spacerRow4, 8, 1 ); + + TQLabel* dbNameText = new TQLabel( i18n( "Database name:" ), this ); + dbNameText->setFixedSize( TQSize( 100, 20 ) ); + dbNameText->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + layout->addWidget( dbNameText, 9, 1 ); + + dbNameEdit = new KLineEdit( this ); + dbNameEdit->setFixedSize( TQSize( 120, 20 ) ); + dbNameEdit->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + layout->addWidget( dbNameEdit, 9, 2 ); + + TQSpacerItem* spacerRow5 = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::MinimumExpanding ); + layout->addItem( spacerRow5, 10, 1 ); + + // Backup options + TQGroupBox *backupGBox = new TQGroupBox( this, "backupGBox" ); + backupGBox->setTitle( i18n( "Backup" ) ); + backupGBox->setColumns( 2 ); + layout->addMultiCellWidget( backupGBox, 10, 10, 1, 4 ); + + TQLabel *dumpPathLabel = new TQLabel( backupGBox ); + dumpPathLabel->setText( TQString(i18n( "Path to '%1':" )).arg("mysqldump") ); + dumpPathRequester = new KURLRequester( backupGBox ); + dumpPathRequester->setFilter( "mysqldump" ); + + TQLabel *mysqlPathLabel = new TQLabel( backupGBox ); + mysqlPathLabel->setText( TQString(i18n( "Path to '%1':" )).arg("mysql") ); + mysqlPathRequester = new KURLRequester( backupGBox ); + mysqlPathRequester->setFilter( "mysql" ); + + + TQSpacerItem* spacerRow6 = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::MinimumExpanding ); + layout->addItem( spacerRow6, 11, 1 ); + TQSpacerItem* spacerRight = new TQSpacerItem( 10, 10, TQSizePolicy::MinimumExpanding, TQSizePolicy::Fixed ); + layout->addItem( spacerRight, 1, 4 ); + + // Load & Save Current Settings + TDEConfig *config = kapp->config(); + config->setGroup( "Server" ); + serverEdit->setText( config->readEntry( "Host", "localhost" ) ); + usernameEdit->setText( config->readEntry( "Username", "" ) ); + passwordEdit->setText( config->readEntry( "Password", "" ) ); + portEdit->setValue( config->readNumEntry( "Port", 0 ) ); + dbNameEdit->setText( config->readEntry( "DBName", "Krecipes" ) ); + dumpPathRequester->setURL( config->readEntry( "MySQLDumpPath", "mysqldump" ) ); + mysqlPathRequester->setURL( config->readEntry( "MySQLPath", "mysql" ) ); +} + +void MySQLServerPrefs::saveOptions( void ) +{ + TDEConfig * config = kapp->config(); + config->setGroup( "Server" ); + config->writeEntry( "Host", serverEdit->text() ); + config->writeEntry( "Username", usernameEdit->text() ); + config->writeEntry( "Password", passwordEdit->text() ); + config->writeEntry( "Port", portEdit->value() ); + config->writeEntry( "DBName", dbNameEdit->text() ); + config->writeEntry( "MySQLDumpPath", dumpPathRequester->url() ); + config->writeEntry( "MySQLPath", mysqlPathRequester->url() ); +} + + +PostgreSQLServerPrefs::PostgreSQLServerPrefs( TQWidget *parent ) : TQWidget( parent ) +{ + TQGridLayout * layout = new TQGridLayout( this, 1, 1, 0, 0 ); + layout->setSpacing( KDialog::spacingHint() ); + layout->setMargin( 0 ); + + TQSpacerItem* spacerTop = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( spacerTop, 0, 1 ); + TQSpacerItem* spacerLeft = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::Minimum ); + layout->addItem( spacerLeft, 1, 0 ); + + TQLabel* serverText = new TQLabel( i18n( "Server:" ), this ); + serverText->setFixedSize( TQSize( 100, 20 ) ); + serverText->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + layout->addWidget( serverText, 1, 1 ); + + serverEdit = new KLineEdit( this ); + serverEdit->setFixedSize( TQSize( 120, 20 ) ); + serverEdit->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + layout->addWidget( serverEdit, 1, 2 ); + + TQSpacerItem* spacerRow1 = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( spacerRow1, 2, 1 ); + + TQLabel* usernameText = new TQLabel( i18n( "Username:" ), this ); + usernameText->setFixedSize( TQSize( 100, 20 ) ); + usernameText->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + layout->addWidget( usernameText, 3, 1 ); + + usernameEdit = new KLineEdit( this ); + usernameEdit->setFixedSize( TQSize( 120, 20 ) ); + usernameEdit->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + layout->addWidget( usernameEdit, 3, 2 ); + + TQSpacerItem* spacerRow2 = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( spacerRow2, 4, 1 ); + + TQLabel* passwordText = new TQLabel( i18n( "Password:" ), this ); + passwordText->setFixedSize( TQSize( 100, 20 ) ); + passwordText->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + layout->addWidget( passwordText, 5, 1 ); + + passwordEdit = new KLineEdit( this ); + passwordEdit->setFixedSize( TQSize( 120, 20 ) ); + passwordEdit->setEchoMode( TQLineEdit::Password ); + passwordEdit->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + layout->addWidget( passwordEdit, 5, 2 ); + + TQSpacerItem* spacerRow3 = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( spacerRow3, 6, 1 ); + + TQLabel* portText = new TQLabel( i18n( "Port:" ), this ); + portText->setFixedSize( TQSize( 100, 20 ) ); + portText->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + layout->addWidget( portText, 7, 1 ); + + portEdit = new KIntNumInput( this ); + portEdit->setMinValue(0); + portEdit->setSpecialValueText( i18n("Default") ); + portEdit->setFixedSize( TQSize( 120, 20 ) ); + portEdit->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + layout->addWidget( portEdit, 7, 2 ); + + TQSpacerItem* spacerRow4 = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( spacerRow4, 8, 1 ); + + TQLabel* dbNameText = new TQLabel( i18n( "Database name:" ), this ); + dbNameText->setFixedSize( TQSize( 100, 20 ) ); + dbNameText->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + layout->addWidget( dbNameText, 9, 1 ); + + dbNameEdit = new KLineEdit( this ); + dbNameEdit->setFixedSize( TQSize( 120, 20 ) ); + dbNameEdit->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + layout->addWidget( dbNameEdit, 9, 2 ); + + TQSpacerItem* spacerRow5 = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::MinimumExpanding ); + layout->addItem( spacerRow5, 10, 1 ); + + // Backup options + TQGroupBox *backupGBox = new TQGroupBox( this, "backupGBox" ); + backupGBox->setTitle( i18n( "Backup" ) ); + backupGBox->setColumns( 2 ); + layout->addMultiCellWidget( backupGBox, 10, 10, 1, 4 ); + + TQLabel *dumpPathLabel = new TQLabel( backupGBox ); + dumpPathLabel->setText( TQString(i18n( "Path to '%1':" )).arg("pg_dump") ); + dumpPathRequester = new KURLRequester( backupGBox ); + dumpPathRequester->setFilter( "pg_dump" ); + + TQLabel *psqlPathLabel = new TQLabel( backupGBox ); + psqlPathLabel->setText( TQString(i18n( "Path to '%1':" )).arg("psql") ); + psqlPathRequester = new KURLRequester( backupGBox ); + psqlPathRequester->setFilter( "psql" ); + + + TQSpacerItem* spacerRow6 = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::MinimumExpanding ); + layout->addItem( spacerRow6, 11, 1 ); + TQSpacerItem* spacerRight = new TQSpacerItem( 10, 10, TQSizePolicy::MinimumExpanding, TQSizePolicy::Fixed ); + layout->addItem( spacerRight, 1, 4 ); + + // Load & Save Current Settings + TDEConfig *config = kapp->config(); + config->setGroup( "Server" ); + serverEdit->setText( config->readEntry( "Host", "localhost" ) ); + usernameEdit->setText( config->readEntry( "Username", "" ) ); + passwordEdit->setText( config->readEntry( "Password", "" ) ); + portEdit->setValue( config->readNumEntry( "Port", 0 ) ); + dbNameEdit->setText( config->readEntry( "DBName", "Krecipes" ) ); + dumpPathRequester->setURL( config->readEntry( "PgDumpPath", "pg_dump" ) ); + psqlPathRequester->setURL( config->readEntry( "PsqlPath", "psql" ) ); +} + +void PostgreSQLServerPrefs::saveOptions( void ) +{ + TDEConfig * config = kapp->config(); + config->setGroup( "Server" ); + config->writeEntry( "Host", serverEdit->text() ); + config->writeEntry( "Username", usernameEdit->text() ); + config->writeEntry( "Password", passwordEdit->text() ); + config->writeEntry( "Port", portEdit->value() ); + config->writeEntry( "DBName", dbNameEdit->text() ); + config->writeEntry( "PgDumpPath", dumpPathRequester->url() ); + config->writeEntry( "PsqlPath", psqlPathRequester->url() ); +} + + + +SQLiteServerPrefs::SQLiteServerPrefs( TQWidget *parent ) : TQWidget( parent ) +{ + TQVBoxLayout * Form1Layout = new TQVBoxLayout( this ); + + TQHBox *hbox = new TQHBox( this ); + ( void ) new TQLabel( i18n( "Database file:" ), hbox ); + + fileRequester = new KURLRequester( hbox ); + hbox->setStretchFactor( fileRequester, 2 ); + + Form1Layout->addWidget( hbox ); + + TQSpacerItem* spacerRow5 = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::MinimumExpanding ); + Form1Layout->addItem( spacerRow5 ); + + TQString sqliteBinary; + #if HAVE_SQLITE3 + sqliteBinary = "sqlite3"; + #elif HAVE_SQLITE + sqliteBinary = "sqlite"; + #endif + + // Backup options + TQGroupBox *backupGBox = new TQGroupBox( this, "backupGBox" ); + backupGBox->setTitle( i18n( "Backup" ) ); + backupGBox->setColumns( 2 ); + Form1Layout->addWidget( backupGBox ); + + TQLabel *dumpPathLabel = new TQLabel( backupGBox ); + dumpPathLabel->setText( TQString(i18n( "Path to '%1':" )).arg(sqliteBinary) ); + dumpPathRequester = new KURLRequester( backupGBox ); + dumpPathRequester->setFilter( sqliteBinary ); + + // Load & Save Current Settings + TDEConfig *config = kapp->config(); + config->setGroup( "Server" ); + fileRequester->setURL( config->readEntry( "DBFile", locateLocal( "appdata", "krecipes.krecdb" ) ) ); + dumpPathRequester->setURL( config->readEntry( "SQLitePath", sqliteBinary ) ); +} + +void SQLiteServerPrefs::saveOptions( void ) +{ + TDEConfig * config = kapp->config(); + config->setGroup( "Server" ); + config->writeEntry( "DBFile", fileRequester->url() ); + config->writeEntry( "SQLitePath", dumpPathRequester->url() ); +} + + +// Server Setttings Dialog +ServerPrefs::ServerPrefs( TQWidget *parent ) + : TQWidget( parent ) +{ + TQVBoxLayout * Form1Layout = new TQVBoxLayout( this, 11, 6 ); + + TDEConfig *config = kapp->config(); + config->setGroup( "DBType" ); + TQString DBtype = config->readEntry( "Type" ); + if ( DBtype == "MySQL" ) + serverWidget = new MySQLServerPrefs( this ); + else if ( DBtype == "PostgreSQL" ) + serverWidget = new PostgreSQLServerPrefs( this ); + else + serverWidget = new SQLiteServerPrefs( this ); + + serverWidget->setSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::Minimum ); + Form1Layout->addWidget( serverWidget ); + + Form1Layout->addItem( new TQSpacerItem( 20, 40, TQSizePolicy::Minimum, TQSizePolicy::Expanding ) ); + + wizard_button = new TQCheckBox( i18n( "Re-run wizard on next startup" ), this ); + wizard_button->setSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::Minimum ); + Form1Layout->addWidget( wizard_button ); + + TQLabel *note = new TQLabel( i18n( "Note: Krecipes must be restarted for most server preferences to take effect." ), this ); + Form1Layout->addWidget( note ); + + adjustSize(); +} + + +void KrecipesPreferences::saveSettings( void ) +{ + m_pageServer->saveOptions(); + m_pageNumbers->saveOptions(); + m_pageImport->saveOptions(); + m_pagePerformance->saveOptions(); +} + +// Save Server settings +void ServerPrefs::saveOptions( void ) +{ + TDEConfig * config = kapp->config(); + config->setGroup( "DBType" ); + TQString DBtype = config->readEntry( "Type" ); + if ( DBtype == "MySQL" ) + ( ( MySQLServerPrefs* ) serverWidget ) ->saveOptions(); + else if ( DBtype == "PostgreSQL" ) + ( ( PostgreSQLServerPrefs* ) serverWidget ) ->saveOptions(); + else + ( ( SQLiteServerPrefs* ) serverWidget ) ->saveOptions(); + + if ( wizard_button->isChecked() ) { + config->setGroup( "Wizard" ); + config->writeEntry( "SystemSetup", false ); + } +} + +//=============Numbers Preferences Dialog================// +NumbersPrefs::NumbersPrefs( TQWidget *parent ) + : TQWidget( parent ) +{ + Form1Layout = new TQVBoxLayout( this, 11, 6 ); + + numberButtonGroup = new TQButtonGroup( this ); + numberButtonGroup->setColumnLayout( 0, TQt::Vertical ); + numberButtonGroup->layout() ->setSpacing( 6 ); + numberButtonGroup->layout() ->setMargin( 11 ); + numberButtonGroup->resize( TQSize() ); + numberButtonGroupLayout = new TQVBoxLayout( numberButtonGroup->layout() ); + numberButtonGroupLayout->setAlignment( TQt::AlignTop ); + + fractionRadioButton = new TQRadioButton( numberButtonGroup ); + numberButtonGroupLayout->addWidget( fractionRadioButton ); + + decimalRadioButton = new TQRadioButton( numberButtonGroup ); + numberButtonGroupLayout->addWidget( decimalRadioButton ); + Form1Layout->addWidget( numberButtonGroup ); + + numberButtonGroup->insert( decimalRadioButton, 0 ); + numberButtonGroup->insert( fractionRadioButton, 1 ); + + //ingredient display format + TQGroupBox *ingredientGrpBox = new TQGroupBox( 2, TQt::Vertical, i18n( "Ingredients" ), this ); + + TQHBox *ingredientBox = new TQHBox( ingredientGrpBox ); + ( void ) new TQLabel( i18n( "Ingredient Format:" ), ingredientBox ); + ingredientEdit = new KLineEdit( ingredientBox ); + ( void ) new TQLabel( i18n( "%n: Name<br>" + "%p: Preparation method<br>" + "%a: Amount<br>" + "%u: Unit" + ), ingredientGrpBox ); + + Form1Layout->addWidget( ingredientGrpBox ); + + //unit display format + TQGroupBox *abbrevGrpBox = new TQGroupBox( 1, TQt::Vertical, i18n( "Units" ), this ); + TQHBox *abbrevBox = new TQHBox( abbrevGrpBox ); + abbrevButton = new TQCheckBox( i18n( "Use abbreviations" ), abbrevBox ); + Form1Layout->addWidget( abbrevGrpBox ); + + + Form1Layout->addItem( new TQSpacerItem( 20, 40, TQSizePolicy::Minimum, TQSizePolicy::Expanding ) ); + + adjustSize(); + + languageChange(); + + // Load Current Settings + TDEConfig *config = kapp->config(); + config->setGroup( "Formatting" ); + + int button = ( config->readBoolEntry( "Fraction", false ) ) ? 1 : 0; + numberButtonGroup->setButton( button ); + + ingredientEdit->setText( config->readEntry( "Ingredient", "%n%p: %a %u" ) ); + + abbrevButton->setChecked( config->readBoolEntry( "AbbreviateUnits", false ) ); +} + +void NumbersPrefs::saveOptions() +{ + TDEConfig * config = kapp->config(); + config->setGroup( "Formatting" ); + + bool fraction = !numberButtonGroup->find( 0 ) ->isOn(); + config->writeEntry( "Fraction", fraction ); + + config->writeEntry( "Ingredient", ingredientEdit->text() ); + + config->writeEntry( "AbbreviateUnits", abbrevButton->isChecked() ); +} + +void NumbersPrefs::languageChange() +{ + numberButtonGroup->setTitle( i18n( "Number Format" ) ); + fractionRadioButton->setText( i18n( "Fraction" ) ); + decimalRadioButton->setText( i18n( "Decimal" ) ); +} + +//=============Import/Export Preferences Dialog================// +ImportPrefs::ImportPrefs( TQWidget *parent ) + : TQWidget( parent ) +{ + // Load Current Settings + TDEConfig * config = kapp->config(); + config->setGroup( "Import" ); + + bool overwrite = config->readBoolEntry( "OverwriteExisting", false ); + bool direct = config->readBoolEntry( "DirectImport", false ); + + Form1Layout = new TQVBoxLayout( this, 11, 6 ); + + TQGroupBox *importGroup = new TQGroupBox(2,TQt::Vertical,i18n("Import"), this); + + overwriteCheckbox = new TQCheckBox( i18n( "Overwrite recipes with same title" ), importGroup ); + overwriteCheckbox->setChecked( overwrite ); + overwriteCheckbox->setSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::Minimum ); + + directImportCheckbox = new TQCheckBox( i18n( "Ask which recipes to import" ), importGroup ); + directImportCheckbox->setChecked( !direct ); + directImportCheckbox->setSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::Minimum ); + + Form1Layout->addWidget(importGroup); + + TQGroupBox *exportGroup = new TQGroupBox(1,TQt::Vertical,i18n("Export"), this); + + TQHBox *clipboardHBox = new TQHBox(exportGroup); + clipboardHBox->setSpacing(6); + TQLabel *clipboardLabel = new TQLabel(i18n("'Copy to Clipboard' format:"),clipboardHBox); + clipBoardFormatComboBox = new TQComboBox( clipboardHBox ); + clipBoardFormatComboBox->insertItem(TQString("%3 (*.txt)").arg(i18n("Plain Text"))); + clipBoardFormatComboBox->insertItem("Krecipes (*.kreml)"); + clipBoardFormatComboBox->insertItem("Meal-Master (*.mmf)"); + clipBoardFormatComboBox->insertItem("Rezkonv (*.rk)"); + clipBoardFormatComboBox->insertItem("RecipeML (*.xml)"); + //clipBoardFormatComboBox->insertItem("CookML (*.cml)"); + clipboardHBox->setStretchFactor(clipBoardFormatComboBox,1); + + config->setGroup( "Export" ); + TQString clipboardFormat = config->readEntry("ClipboardFormat"); + if ( clipboardFormat == "*.kreml" ) + clipBoardFormatComboBox->setCurrentItem(1); + else if ( clipboardFormat == "*.mmf" ) + clipBoardFormatComboBox->setCurrentItem(2); + else if ( clipboardFormat == "*.xml" ) + clipBoardFormatComboBox->setCurrentItem(3); + else + clipBoardFormatComboBox->setCurrentItem(0); + + Form1Layout->addWidget(exportGroup); + + Form1Layout->addItem( new TQSpacerItem( 20, 40, TQSizePolicy::Minimum, TQSizePolicy::Expanding ) ); + + TQWhatsThis::add( directImportCheckbox, + i18n("When this is enabled, the importer will show every recipe in the file(s) and allow you to select which recipes you want imported.\n \ + \ + Disable this to always import every recipe, which allows for faster and less memory-intensive imports.") + ); + + adjustSize(); +} + +void ImportPrefs::saveOptions() +{ + TDEConfig * config = kapp->config(); + config->setGroup( "Import" ); + + config->writeEntry( "OverwriteExisting", overwriteCheckbox->isChecked() ); + config->writeEntry( "DirectImport", !directImportCheckbox->isChecked() ); + + config->setGroup( "Export" ); + TQString ext = clipBoardFormatComboBox->currentText().mid(clipBoardFormatComboBox->currentText().find("(")+1,clipBoardFormatComboBox->currentText().length()-clipBoardFormatComboBox->currentText().find("(")-2); + config->writeEntry( "ClipboardFormat", ext ); +} + + +//=============Performance Options Dialog================// +PerformancePrefs::PerformancePrefs( TQWidget *parent ) + : TQWidget( parent ) +{ + // Load Current Settings + TDEConfig * config = kapp->config(); + config->setGroup( "Performance" ); + + int cat_limit = config->readNumEntry( "CategoryLimit", -1 ); + int limit = config->readNumEntry( "Limit", -1 ); + + Form1Layout = new TQVBoxLayout( this, 11, 6 ); + + searchAsYouTypeBox = new TQCheckBox( i18n( "Search as you type" ), this ); + searchAsYouTypeBox->setChecked( config->readBoolEntry( "SearchAsYouType", true ) ); + + TQLabel *explainationLabel = new TQLabel( i18n("In most instances these options do not need to be changed. However, limiting the amount of items displayed at once will <b>allow Krecipes to better perform when the database is loaded with many thousands of recipes</b>."), this ); + explainationLabel->setTextFormat( TQt::RichText ); + + TQHBox *catLimitHBox = new TQHBox( this ); + catLimitInput = new KIntNumInput(catLimitHBox); + catLimitInput->setLabel( i18n( "Number of categories to display at once:" ) ); + catLimitInput->setRange(0,5000,20,true); + catLimitInput->setSpecialValueText( i18n("Unlimited") ); + + if ( cat_limit > 0 ) + catLimitInput->setValue( cat_limit ); + + TQHBox *limitHBox = new TQHBox( this ); + limitInput = new KIntNumInput(limitHBox); + limitInput->setLabel( i18n( "Number of elements to display at once:" ) ); + limitInput->setRange(0,100000,1000,true); + limitInput->setSpecialValueText( i18n("Unlimited") ); + + if ( limit > 0 ) + limitInput->setValue( limit ); + + Form1Layout->addWidget( searchAsYouTypeBox ); + Form1Layout->addWidget( explainationLabel ); + Form1Layout->addWidget( catLimitHBox ); + Form1Layout->addWidget( limitHBox ); + + Form1Layout->addItem( new TQSpacerItem( 20, 40, TQSizePolicy::Minimum, TQSizePolicy::Expanding ) ); + + adjustSize(); +} + +void PerformancePrefs::saveOptions() +{ + TDEConfig * config = kapp->config(); + config->setGroup( "Performance" ); + + int catLimit = ( catLimitInput->value() == 0 ) ? -1 : catLimitInput->value(); + config->writeEntry( "CategoryLimit", catLimit ); + + int limit = ( limitInput->value() == 0 ) ? -1 : limitInput->value(); + config->writeEntry( "Limit", limit ); + + config->writeEntry( "SearchAsYouType", searchAsYouTypeBox->isChecked() ); +} + +#include "pref.moc" diff --git a/src/pref.h b/src/pref.h new file mode 100644 index 0000000..31b7707 --- /dev/null +++ b/src/pref.h @@ -0,0 +1,182 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* Copyright (C) 2006 Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef _KRECIPESPREF_H_ +#define _KRECIPESPREF_H_ + +#include <kdialogbase.h> + +#include <tqmap.h> + +class ServerPrefs; +class NumbersPrefs; +class ImportPrefs; +class PerformancePrefs; + +class KIntNumInput; +class KLineEdit; +class KURLRequester; + +class TQButtonGroup; +class TQCheckBox; +class TQRadioButton; +class TQVBoxLayout; +class TQComboBox; + +class KrecipesPreferences : public KDialogBase +{ + TQ_OBJECT +public: + KrecipesPreferences( TQWidget *parent ); + +protected slots: + void slotHelp(); + +private: + ServerPrefs *m_pageServer; + NumbersPrefs *m_pageNumbers; + ImportPrefs *m_pageImport; + PerformancePrefs *m_pagePerformance; + + TQMap<int,TQString> m_helpMap; + +private slots: + void saveSettings( void ); +}; + + +class MySQLServerPrefs : public TQWidget +{ +public: + MySQLServerPrefs( TQWidget *parent ); + + void saveOptions( void ); +private: + // Internal Widgets + KURLRequester *dumpPathRequester; + KURLRequester *mysqlPathRequester; + + KLineEdit *serverEdit; + KLineEdit *usernameEdit; + KLineEdit *passwordEdit; + KLineEdit *dbNameEdit; + KIntNumInput *portEdit; +}; + +class PostgreSQLServerPrefs : public TQWidget +{ +public: + PostgreSQLServerPrefs( TQWidget *parent ); + + void saveOptions( void ); +private: + // Internal Widgets + KURLRequester *dumpPathRequester; + KURLRequester *psqlPathRequester; + + KLineEdit *serverEdit; + KLineEdit *usernameEdit; + KLineEdit *passwordEdit; + KLineEdit *dbNameEdit; + KIntNumInput *portEdit; +}; + +class SQLiteServerPrefs : public TQWidget +{ + TQ_OBJECT + +public: + SQLiteServerPrefs( TQWidget *parent ); + + void saveOptions( void ); + +private: + // Internal Widgets + KURLRequester *dumpPathRequester; + KURLRequester *fileRequester; +}; + + +class ServerPrefs : public TQWidget +{ + TQ_OBJECT +public: + ServerPrefs( TQWidget *parent = 0 ); + + // Public Methods + void saveOptions( void ); +private: + TQWidget *serverWidget; + TQCheckBox *wizard_button; +}; + +class NumbersPrefs : public TQWidget +{ + TQ_OBJECT + +public: + NumbersPrefs( TQWidget *parent = 0 ); + + void saveOptions(); + +protected: + TQButtonGroup* numberButtonGroup; + TQRadioButton* fractionRadioButton; + TQRadioButton* decimalRadioButton; + + TQVBoxLayout* Form1Layout; + TQVBoxLayout* numberButtonGroupLayout; + + KLineEdit *ingredientEdit; + TQCheckBox *abbrevButton; + +protected slots: + virtual void languageChange(); +}; + +class ImportPrefs : public TQWidget +{ + TQ_OBJECT + +public: + ImportPrefs( TQWidget *parent = 0 ); + + void saveOptions(); + +protected: + TQVBoxLayout* Form1Layout; + TQCheckBox* overwriteCheckbox; + TQCheckBox* directImportCheckbox; + + TQComboBox *clipBoardFormatComboBox; +}; + + +class PerformancePrefs : public TQWidget +{ + TQ_OBJECT + +public: + PerformancePrefs( TQWidget *parent = 0 ); + + void saveOptions(); + +protected: + TQVBoxLayout* Form1Layout; + TQCheckBox* searchAsYouTypeBox; + KIntNumInput* catLimitInput; + KIntNumInput* limitInput; +}; + +#endif // _KRECIPESPREF_H_ diff --git a/src/profiling.h b/src/profiling.h new file mode 100644 index 0000000..f42ac70 --- /dev/null +++ b/src/profiling.h @@ -0,0 +1,34 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef PROFILING_H +#define PROFILING_H + +#ifdef HAVE_CONFIG_H + #include "config.h" +#endif + +#define KRECIPES_PROFILING + +#ifdef KRECIPES_PROFILING + #include <tqdatetime.h> + #include <kdebug.h> + static TQTime dbg_timer; + + #define START_TIMER(MSG) \ + dbg_timer.start(); kdDebug()<<MSG<<endl; + #define END_TIMER() \ + kdDebug()<<"...took "<<dbg_timer.elapsed()<<" ms"<<endl; +#else + #define START_TIMER(MSG) + #define END_TIMER() +#endif + +#endif //PROFILING_H diff --git a/src/propertycalculator.cpp b/src/propertycalculator.cpp new file mode 100644 index 0000000..8310b14 --- /dev/null +++ b/src/propertycalculator.cpp @@ -0,0 +1,131 @@ +/*************************************************************************** +* Copyright (C) 2003 by Unai Garro * +* [email protected] * +* * +* 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. * +***************************************************************************/ +#include "propertycalculator.h" + +#include <math.h> // For fabs() + +#include <kdebug.h> + +#include "backends/recipedb.h" +#include "datablocks/elementlist.h" +#include "datablocks/ingredientpropertylist.h" +#include "datablocks/recipe.h" + +bool autoConvert( RecipeDB* database, const Ingredient &from, const Ingredient &to, Ingredient &result ) +{ + RecipeDB::ConversionStatus status = database->convertIngredientUnits( from, to.units, result ); + bool converted = status == RecipeDB::Success || status == RecipeDB::MismatchedPrepMethodUsingApprox; + + if ( converted ) // There is a ratio + { + double ratio = result.amount / from.amount; + + if ( ratio > 1 ) // Convert to unit 1, since unit1 is bigger + { + result.units = from.units; + result.amount = from.amount + to.amount / ratio; + } + else { //Convert to unit2, since unit2 is bigger (just add, units are now correct) + result.amount += to.amount; + } + return true; + } + else + return false; +} + +/* +** Version with database I/O. DB must be provided +*/ + +void calculateProperties( Recipe& recipe, RecipeDB* database ) +{ + recipe.properties.clear(); + // Note that recipePropertyList is not attached to any ingredient. It's just the total of the recipe + IngredientPropertyList ingredientPropertyList; // property list for each ingredient + + int ingredientNo = 1; + + for ( IngredientList::const_iterator ing_it = recipe.ingList.begin(); ing_it != recipe.ingList.end(); ++ing_it ) { + database->loadProperties( &ingredientPropertyList, ( *ing_it ).ingredientID ); + ingredientPropertyList.divide( recipe.yield.amount ); // calculates properties per yield unit + addPropertyToList( database, &recipe.properties, ingredientPropertyList, *ing_it, ingredientNo ); + ingredientNo++; + } +} + + +void addPropertyToList( RecipeDB *database, IngredientPropertyList *recipePropertyList, IngredientPropertyList &ingPropertyList, const Ingredient &ing, int ingredientNo ) +{ + TQMap<int,double> ratioCache; //unit->ratio + + IngredientPropertyList::const_iterator prop_it; + for ( prop_it = ingPropertyList.begin(); prop_it != ingPropertyList.end(); ++prop_it ) { + // Find if property was listed before + int pos = recipePropertyList->findIndex( *prop_it ); + if ( pos >= 0 ) //Exists. Add to it + { + IngredientPropertyList::iterator rec_property_it = recipePropertyList->at( pos ); + Ingredient result; + + bool converted; + TQMap<int,double>::const_iterator cache_it = ratioCache.find((*prop_it).perUnit.id); + if ( cache_it == ratioCache.end() ) { + RecipeDB::ConversionStatus status = database->convertIngredientUnits( ing, (*prop_it).perUnit, result ); + converted = status == RecipeDB::Success || status == RecipeDB::MismatchedPrepMethodUsingApprox; + + if ( converted ) + ratioCache.insert((*prop_it).perUnit.id,result.amount / ing.amount); + else + ratioCache.insert((*prop_it).perUnit.id,-1); + } + else { + result.units = (*prop_it).perUnit; + result.amount = ing.amount * (*cache_it); + converted = result.amount > 0; + } + + if ( converted ) // Could convert units to perUnit + (*rec_property_it).amount += ( (*prop_it).amount ) * result.amount; + } + else // Append new property + { + IngredientProperty property; + property.id = (*prop_it).id; + property.name = (*prop_it).name; + property.perUnit.id = -1; // It's not per unit, it's total sum of the recipe + property.perUnit.name = TQString::null; // " + property.units = (*prop_it).units; + + Ingredient result; + bool converted; + TQMap<int,double>::const_iterator cache_it = ratioCache.find((*prop_it).perUnit.id); + if ( cache_it == ratioCache.end() ) { + RecipeDB::ConversionStatus status = database->convertIngredientUnits( ing, (*prop_it).perUnit, result ); + converted = status == RecipeDB::Success || status == RecipeDB::MismatchedPrepMethodUsingApprox; + if ( converted ) + ratioCache.insert((*prop_it).perUnit.id,result.amount / ing.amount); + else + ratioCache.insert((*prop_it).perUnit.id,-1); + } + else { + result.units = (*prop_it).perUnit; + result.amount = ing.amount * (*cache_it); + converted = result.amount > 0; + } + + if ( converted ) // Could convert units to perUnit + { + property.amount = ( (*prop_it).amount ) * result.amount; + recipePropertyList->append( property ); + } + } + } +} diff --git a/src/propertycalculator.h b/src/propertycalculator.h new file mode 100644 index 0000000..3553d52 --- /dev/null +++ b/src/propertycalculator.h @@ -0,0 +1,24 @@ +/*************************************************************************** +* Copyright (C) 2003 by Unai Garro * +* [email protected] * +* * +* 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. * +***************************************************************************/ + +#ifndef PROPERTYCALCULATOR_H +#define PROPERTYCALCULATOR_H + +class RecipeDB; +class Ingredient; +class IngredientPropertyList; +class Recipe; + +bool autoConvert( RecipeDB* database, const Ingredient &from, const Ingredient &to, Ingredient &result ); +void checkUndefined( IngredientPropertyList *recipePropertyList, IngredientPropertyList &addedPropertyList ); +void calculateProperties( Recipe& recipe, RecipeDB* database ); +void addPropertyToList( RecipeDB *database, IngredientPropertyList *recipePropertyList, IngredientPropertyList &ingPropertyList, const Ingredient &ing, int ingredientNo ); + +#endif //PROPERTYCALCULATOR_H diff --git a/src/recipeactionshandler.cpp b/src/recipeactionshandler.cpp new file mode 100644 index 0000000..dde3fcb --- /dev/null +++ b/src/recipeactionshandler.cpp @@ -0,0 +1,475 @@ +/*************************************************************************** +* Copyright (C) 2003-2004 by * +* Jason Kivlighn ([email protected]) * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "recipeactionshandler.h" + +#include <tqwidget.h> +#include <tqclipboard.h> + +#include <tdeapplication.h> +#include <tdefiledialog.h> +#include <kiconloader.h> +#include <tdelistview.h> +#include <tdelocale.h> +#include <tdemessagebox.h> +#include <tdepopupmenu.h> +#include <kprogress.h> + +#include "dialogs/selectcategoriesdialog.h" + +#include "exporters/cookmlexporter.h" +#include "exporters/htmlexporter.h" +#include "exporters/htmlbookexporter.h" +#include "exporters/kreexporter.h" +#include "exporters/mmfexporter.h" +#include "exporters/recipemlexporter.h" +#include "exporters/plaintextexporter.h" +#include "exporters/rezkonvexporter.h" + +#include "widgets/recipelistview.h" +#include "widgets/categorylistview.h" + +#include "backends/recipedb.h" + +RecipeActionsHandler::RecipeActionsHandler( TDEListView *_parentListView, RecipeDB *db, int actions ) : TQObject( _parentListView ), + parentListView( _parentListView ), + database( db ) +{ + TDEIconLoader * il = new TDEIconLoader; + + kpop = new TDEPopupMenu( parentListView ); + if ( actions & Open ) + kpop->insertItem( il->loadIcon( "ok", TDEIcon::NoGroup, 16 ), i18n( "&Open" ), this, TQ_SLOT( open() ), CTRL + Key_L ); + if ( actions & Edit ) + kpop->insertItem( il->loadIcon( "edit", TDEIcon::NoGroup, 16 ), i18n( "&Edit" ), this, TQ_SLOT( edit() ), CTRL + Key_E ); + if ( actions & Export ) + kpop->insertItem( il->loadIcon( "fileexport", TDEIcon::NoGroup, 16 ), i18n( "E&xport" ), this, TQ_SLOT( recipeExport() ), 0 ); + if ( actions & RemoveFromCategory ) + remove_from_cat_item = kpop->insertItem( il->loadIcon( "editshred", TDEIcon::NoGroup, 16 ), i18n( "&Remove From Category" ), this, TQ_SLOT( removeFromCategory() ), CTRL + Key_R ); + if ( actions & Remove ) + kpop->insertItem( il->loadIcon( "editshred", TDEIcon::NoGroup, 16 ), i18n( "&Delete" ), this, TQ_SLOT( remove + () ), Key_Delete ); + if ( actions & AddToShoppingList ) + kpop->insertItem( il->loadIcon( "trolley", TDEIcon::NoGroup, 16 ), i18n( "&Add to Shopping List" ), this, TQ_SLOT( addToShoppingList() ), CTRL + Key_A ); + if ( actions & CopyToClipboard ) + kpop->insertItem( il->loadIcon( "edit-copy", TDEIcon::NoGroup, 16 ), i18n( "&Copy to Clipboard" ), this, TQ_SLOT( recipesToClipboard() ), CTRL + Key_C ); + + if ( actions & Categorize ) + categorize_item = kpop->insertItem( il->loadIcon( "categories", TDEIcon::NoGroup, 16 ), i18n( "Ca&tegorize..." ), this, TQ_SLOT(categorize()), CTRL + Key_T ); + + kpop->polish(); + + catPop = new TDEPopupMenu( parentListView ); + if ( actions & ExpandAll ) + catPop->insertItem( i18n( "&Expand All" ), this, TQ_SLOT( expandAll() ), CTRL + Key_Plus ); + if ( actions & CollapseAll ) + catPop->insertItem( i18n( "&Collapse All" ), this, TQ_SLOT( collapseAll() ), CTRL + Key_Minus ); + if ( actions & Export ) + catPop->insertItem( il->loadIcon( "fileexport", TDEIcon::NoGroup, 16 ), i18n( "E&xport" ), this, TQ_SLOT( recipeExport() ), 0 ); + + catPop->polish(); + + delete il; + + connect( parentListView, TQ_SIGNAL( contextMenu( TDEListView *, TQListViewItem *, const TQPoint & ) ), TQ_SLOT( showPopup( TDEListView *, TQListViewItem *, const TQPoint & ) ) ); + connect( parentListView, TQ_SIGNAL( doubleClicked( TQListViewItem*, const TQPoint &, int ) ), TQ_SLOT( open() ) ); +} + +void RecipeActionsHandler::exec( ItemType type, const TQPoint &p ) +{ + if ( type == Recipe ) { + if ( kpop->idAt( 0 ) != -1 ) + kpop->exec( p ); + } + else if ( type == Category ) { + if ( catPop->idAt( 0 ) != -1 ) + catPop->exec( p ); + } +} + +void RecipeActionsHandler::showPopup( TDEListView * /*l*/, TQListViewItem *i, const TQPoint &p ) +{ + if ( i ) { // Check if the TQListViewItem actually exists + if ( i->rtti() == 1000 ) { + kpop->setItemVisible( categorize_item, i->parent() && i->parent()->rtti() == 1006 ); + kpop->setItemVisible( remove_from_cat_item, i->parent() && i->parent()->rtti() == 1001 ); + exec( Recipe, p ); + } + else if ( i->rtti() == 1001 ) //is a category... don't pop-up for an empty category though + exec( Category, p ); + } +} + +TQValueList<int> RecipeActionsHandler::recipeIDs( const TQPtrList<TQListViewItem> &items ) const +{ + TQValueList<int> ids; + + TQPtrListIterator<TQListViewItem> it(items); + TQListViewItem *item; + while ( (item = it.current()) != 0 ) { + if ( item->rtti() == 1000 ) { //RecipeListItem + RecipeListItem * recipe_it = ( RecipeListItem* ) item; + if ( ids.find( recipe_it->recipeID() ) == ids.end() ) + ids << recipe_it->recipeID(); + } + else if ( item->rtti() == 1001 ) { + CategoryListItem *cat_it = ( CategoryListItem* ) item; + ElementList list; + database->loadRecipeList( &list, cat_it->element().id, true ); + + for ( ElementList::const_iterator cat_it = list.begin(); cat_it != list.end(); ++cat_it ) { + if ( ids.find( (*cat_it).id ) == ids.end() ) + ids << (*cat_it).id; + } + } + ++it; + } + + return ids; +} + +void RecipeActionsHandler::open() +{ + TQPtrList<TQListViewItem> items = parentListView->selectedItems(); + if ( items.count() > 0 ) { + TQValueList<int> ids = recipeIDs(items); + if ( ids.count() == 1 ) + emit recipeSelected(ids.first(),0); + else if ( ids.count() > 0 ) + emit recipesSelected(ids,0); + #if 0 + else if ( it->rtti() == 1001 && it->firstChild() ) //CategoryListItem and not empty + { + TQValueList<int> ids; + + //do this to only iterate over children of 'it' + TQListViewItem *pEndItem = NULL; + TQListViewItem *pStartItem = it; + do + { + if ( pStartItem->nextSibling() ) + pEndItem = pStartItem->nextSibling(); + else + pStartItem = pStartItem->parent(); + } + while ( pStartItem && !pEndItem ); + + TQListViewItemIterator iterator( it ); + while ( iterator.current() != pEndItem ) + { + if ( iterator.current() ->rtti() == 1000 ) { + RecipeListItem * recipe_it = ( RecipeListItem* ) iterator.current(); + if ( ids.find( recipe_it->recipeID() ) == ids.end() ) { + ids.append( recipe_it->recipeID() ); + } + } + ++iterator; + } + emit recipesSelected( ids, 0 ); + } + #endif + } +} + +void RecipeActionsHandler::categorize() +{ + TQPtrList<TQListViewItem> items = parentListView->selectedItems(); + if ( items.count() > 0 ) { + ElementList categoryList; + SelectCategoriesDialog *editCategoriesDialog = new SelectCategoriesDialog( parentListView, categoryList, database ); + + if ( editCategoriesDialog->exec() == TQDialog::Accepted ) { // user presses Ok + editCategoriesDialog->getSelectedCategories( &categoryList ); // get the category list chosen + + TQPtrListIterator<TQListViewItem> it(items); + TQListViewItem *item; + while ( (item = it.current()) != 0 ) { + if ( item->parent() != 0 ) { + RecipeListItem * recipe_it = ( RecipeListItem* ) item; + int recipe_id = recipe_it->recipeID(); + + database->categorizeRecipe( recipe_id, categoryList ); + } + ++it; + } + } + + delete editCategoriesDialog; + } +} + +void RecipeActionsHandler::edit() +{ + TQPtrList<TQListViewItem> items = parentListView->selectedItems(); + if ( items.count() > 1 ) + KMessageBox::sorry( kapp->mainWidget(), i18n("Please select only one recipe."), i18n("Edit Recipe") ); + else if ( items.count() == 1 && items.at(0)->rtti() == 1000 ) { + RecipeListItem * recipe_it = ( RecipeListItem* ) items.at(0); + emit recipeSelected( recipe_it->recipeID(), 1 ); + } + else //either nothing was selected or a category was selected + KMessageBox::sorry( kapp->mainWidget(), i18n("No recipes selected."), i18n("Edit Recipe") ); +} + +void RecipeActionsHandler::recipeExport() +{ + TQPtrList<TQListViewItem> items = parentListView->selectedItems(); + if ( items.count() > 0 ) { + TQValueList<int> ids = recipeIDs( items ); + + TQString title; + if ( items.count() == 1 && items.at(0)->rtti() == 1000 ) { + RecipeListItem * recipe_it = ( RecipeListItem* ) items.at(0); + title = recipe_it->title(); + } + else + title = i18n( "Recipes" ); + + exportRecipes( ids, i18n( "Export Recipe" ), title, database ); + } + else //if nothing selected, export all visible recipes + { + TQValueList<int> ids = getAllVisibleItems(); + if ( ids.count() > 0 ) { + switch ( KMessageBox::questionYesNo( kapp->mainWidget(), i18n("No recipes are currently selected.\nWould you like to export all recipes in the current view?")) ) + { + case KMessageBox::Yes: + exportRecipes( ids, i18n( "Export Recipes" ), i18n( "Recipes" ), database ); + break; + default: break; + } + } + else + KMessageBox::sorry( kapp->mainWidget(), i18n("No recipes selected."), i18n("Export") ); + } +} + +void RecipeActionsHandler::removeFromCategory() +{ + TQPtrList<TQListViewItem> items = parentListView->selectedItems(); + if ( items.count() > 0 ) { + TQPtrListIterator<TQListViewItem> it(items); + TQListViewItem *item; + while ( (item = it.current()) != 0 ) { + if ( item->parent() != 0 ) { + RecipeListItem * recipe_it = ( RecipeListItem* ) item; + int recipe_id = recipe_it->recipeID(); + + CategoryListItem *cat_it = ( CategoryListItem* ) item->parent(); + database->removeRecipeFromCategory( recipe_id, cat_it->categoryId() ); + } + ++it; + } + } +} + +void RecipeActionsHandler::remove() +{ + TQPtrList<TQListViewItem> items = parentListView->selectedItems(); + if ( items.count() > 0 ) { + TQPtrListIterator<TQListViewItem> it(items); + TQListViewItem *item; + while ( (item = it.current()) != 0 ) { + if ( item->rtti() == RECIPELISTITEM_RTTI ) { + RecipeListItem * recipe_it = ( RecipeListItem* ) item; + emit recipeSelected( recipe_it->recipeID(), 2 ); + } + ++it; + } + } +} + +void RecipeActionsHandler::addToShoppingList() +{ + TQPtrList<TQListViewItem> items = parentListView->selectedItems(); + if ( items.count() > 0 ) { + TQPtrListIterator<TQListViewItem> it(items); + TQListViewItem *item; + while ( (item = it.current()) != 0 ) { + if ( item->parent() != 0 ) { + RecipeListItem * recipe_it = ( RecipeListItem* ) item; + emit recipeSelected( recipe_it->recipeID(), 3 ); + } + ++it; + } + } +} + +void RecipeActionsHandler::expandAll() +{ + TQListViewItemIterator it( parentListView ); + while ( it.current() ) { + TQListViewItem * item = it.current(); + item->setOpen( true ); + ++it; + } +} + +void RecipeActionsHandler::collapseAll() +{ + TQListViewItemIterator it( parentListView ); + while ( it.current() ) { + TQListViewItem * item = it.current(); + item->setOpen( false ); + ++it; + } +} + +void RecipeActionsHandler::exportRecipe( int id, const TQString & caption, const TQString &selection, RecipeDB *db ) +{ + TQValueList<int> ids; + ids.append( id ); + + exportRecipes( ids, caption, selection, db ); +} + +void RecipeActionsHandler::exportRecipes( const TQValueList<int> &ids, const TQString & caption, const TQString &selection, RecipeDB *database ) +{ + KFileDialog * fd = new KFileDialog( TQString::null, + TQString( "*.kre|%1 (*.kre)\n" + "*.kreml|Krecipes (*.kreml)\n" + "*.txt|%3 (*.txt)\n" + //"*.cml|CookML (*.cml)\n" + "*|Web Book\n" + "*.html|%2 (*.html)\n" + "*.mmf|Meal-Master (*.mmf)\n" + "*.xml|RecipeML (*.xml)\n" + "*.rk|Rezkonv (*.rk)" + ).arg( i18n( "Compressed Krecipes format" ) ).arg( i18n( "Web page" ) ).arg( i18n("Plain Text") ), + 0, "export_dlg", true ); + fd->setCaption( caption ); + fd->setOperationMode( KFileDialog::Saving ); + fd->setSelection( selection ); + fd->setMode( KFile::File | KFile::Directory ); + if ( fd->exec() == KFileDialog::Accepted ) { + TQString fileName = fd->selectedFile(); + if ( !fileName.isNull() ) { + BaseExporter * exporter; + if ( fd->currentFilter() == "*.xml" ) + exporter = new RecipeMLExporter( fileName, fd->currentFilter() ); + else if ( fd->currentFilter() == "*.mmf" ) + exporter = new MMFExporter( fileName, fd->currentFilter() ); + else if ( fd->currentFilter() == "*" ) { + CategoryTree *cat_structure = new CategoryTree; + database->loadCategories( cat_structure ); + exporter = new HTMLBookExporter( cat_structure, fd->baseURL().path(), "*.html" ); + } + else if ( fd->currentFilter() == "*.html" ) { + exporter = new HTMLExporter( fileName, fd->currentFilter() ); + } + else if ( fd->currentFilter() == "*.cml" ) + exporter = new CookMLExporter( fileName, fd->currentFilter() ); + else if ( fd->currentFilter() == "*.txt" ) + exporter = new PlainTextExporter( fileName, fd->currentFilter() ); + else if ( fd->currentFilter() == "*.rk" ) + exporter = new RezkonvExporter( fileName, fd->currentFilter() ); + else { + CategoryTree *cat_structure = new CategoryTree; + database->loadCategories( cat_structure ); + exporter = new KreExporter( cat_structure, fileName, fd->currentFilter() ); + } + + int overwrite = -1; + if ( TQFile::exists( exporter->fileName() ) ) { + overwrite = KMessageBox::warningYesNo( 0, TQString( i18n( "File \"%1\" exists. Are you sure you want to overwrite it?" ) ).arg( exporter->fileName() ), i18n( "Saving recipe" ) ); + } + + if ( overwrite == KMessageBox::Yes || overwrite == -1 ) { + KProgressDialog progress_dialog( 0, "export_progress_dialog", TQString::null, i18n( "Saving recipes..." ) ); + exporter->exporter( ids, database, &progress_dialog ); + } + delete exporter; + } + } + delete fd; +} + +void RecipeActionsHandler::recipesToClipboard( const TQValueList<int> &ids, RecipeDB *db ) +{ + TDEConfig *config = TDEGlobal::config(); + config->setGroup("Export"); + TQString formatFilter = config->readEntry("ClipboardFormat"); + + BaseExporter * exporter; + if ( formatFilter == "*.xml" ) + exporter = new RecipeMLExporter( TQString::null, formatFilter ); + else if ( formatFilter == "*.mmf" ) + exporter = new MMFExporter( TQString::null, formatFilter ); + else if ( formatFilter == "*.cml" ) + exporter = new CookMLExporter( TQString::null, formatFilter ); + else if ( formatFilter == "*.rk" ) + exporter = new RezkonvExporter( TQString::null, formatFilter ); + else if ( formatFilter == "*.kre" || formatFilter == "*.kreml" ) { + CategoryTree *cat_structure = new CategoryTree; + db->loadCategories( cat_structure ); + exporter = new KreExporter( cat_structure, TQString::null, formatFilter ); + } + else //default to plain text + exporter = new PlainTextExporter( TQString::null, "*.txt" ); + + RecipeList recipeList; + db->loadRecipes( &recipeList, exporter->supportedItems(), ids ); + + TQString buffer; + TQTextStream stream(buffer,IO_WriteOnly); + exporter->writeStream(stream,recipeList); + + delete exporter; + + TQApplication::clipboard()->setText(buffer); +} + +void RecipeActionsHandler::recipesToClipboard() +{ + TQPtrList<TQListViewItem> items = parentListView->selectedItems(); + if ( items.count() > 0 ) { + TQValueList<int> ids = recipeIDs( items ); + + recipesToClipboard(ids,database); + } +} + +TQValueList<int> RecipeActionsHandler::getAllVisibleItems() +{ + TQValueList<int> ids; + + TQListViewItemIterator iterator( parentListView ); + while ( iterator.current() ) { + if ( iterator.current() ->isVisible() ) { + if ( iterator.current() ->rtti() == RECIPELISTITEM_RTTI ) { + RecipeListItem * recipe_it = ( RecipeListItem* ) iterator.current(); + int recipe_id = recipe_it->recipeID(); + + if ( ids.find( recipe_id ) == ids.end() ) + ids.append( recipe_id ); + } + //it is a category item and isn't populated, so get the unpopulated data from the database + else if ( iterator.current()->rtti() == CATEGORYLISTITEM_RTTI && !iterator.current()->firstChild() ) { + int cat_id = (( CategoryListItem* ) iterator.current())->element().id; + ElementList list; + database->loadRecipeList( &list, cat_id, true ); + + for ( ElementList::const_iterator it = list.begin(); it != list.end(); ++it ) { + if ( ids.find( (*it).id ) == ids.end() ) + ids << (*it).id; + } + } + } + + ++iterator; + } + + return ids; +} + +#include "recipeactionshandler.moc" + diff --git a/src/recipeactionshandler.h b/src/recipeactionshandler.h new file mode 100644 index 0000000..f46e8de --- /dev/null +++ b/src/recipeactionshandler.h @@ -0,0 +1,119 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef RECIPEACTIONSHANDLER_H +#define RECIPEACTIONSHANDLER_H + +#include <tqobject.h> +#include <tqvaluelist.h> +#include <tqptrlist.h> + +class TQListViewItem; +class TDEListView; +class TDEPopupMenu; +class RecipeDB; + +/** @brief A class that centralizes common actions for recipes such as saving and editing. + * + * It acts upon a given TDEListView that is assumed to be a list of recipes. It + * automagically enables this list view with a popup menu for user access to + * the provided actions. + * + * @author Jason Kivlighn + */ +class RecipeActionsHandler : public TQObject +{ + TQ_OBJECT + +public: + enum ItemType { Category, Recipe }; + enum RecipeActions { + AllActions = 0xffff, + Open = 0x0001, + Edit = 0x0002, + Export = 0x0004, + RemoveFromCategory = 0x0008, + Remove = 0x0010, + ExpandAll = 0x0020, + CollapseAll = 0x0040, + AddToShoppingList = 0x0080, + CopyToClipboard = 0x0100, + Categorize = 0x0200 + }; + + RecipeActionsHandler( TDEListView *parentListView, RecipeDB *db, int actions = AllActions ); + ~RecipeActionsHandler() + {} + + static void exportRecipes( const TQValueList<int> &ids, const TQString & caption, const TQString &selection, RecipeDB *db ); + static void exportRecipe( int id, const TQString & caption, const TQString &selection, RecipeDB *db ); + static void recipesToClipboard( const TQValueList<int> &ids, RecipeDB *db ); + +signals: + void recipeSelected( int id, int action ); + void recipesSelected( const TQValueList<int> &ids, int action ); + +public slots: + void exec( ItemType type, const TQPoint &p ); + void showPopup( TDEListView *, TQListViewItem *, const TQPoint & ); + + void categorize(); + + /** Signals an open event (via the recipeSelected() signal) for the recipe(s) currently + * selected in the list view + */ + void open(); + + /** Signals an edit event (via the recipeSelected() signal) for the recipe currently + * selected in the list view + */ + void edit(); + + /** Saves the recipe(s) currently selected in the list view, prompting with a file + * dialog. + */ + void recipeExport(); + + /** Removes the recipe(s) currently selected in the list view from its current category */ + void removeFromCategory(); + + /** Removes the recipe(s) currently selected in the list view from the database */ + void remove + (); + + /** Add the recipe(s) currently selected in the list view to the shopping list dialog */ + void addToShoppingList(); + + /** Expands all items in the list view */ + void expandAll(); + + /** Collapses all items in the list view */ + void collapseAll(); + + void recipesToClipboard(); + +private: + TDEPopupMenu *kpop; + TDEPopupMenu *catPop; + + TDEListView *parentListView; + RecipeDB *database; + + int remove_from_cat_item; + int categorize_item; + + TQValueList<int> getAllVisibleItems(); + TQValueList<int> recipeIDs( const TQPtrList<TQListViewItem> &items ) const; +}; + +#endif //RECIPEACTIONSHANDLER_H + diff --git a/src/recipefilter.cpp b/src/recipefilter.cpp new file mode 100644 index 0000000..fc8d8ee --- /dev/null +++ b/src/recipefilter.cpp @@ -0,0 +1,154 @@ +/*************************************************************************** +* Copyright (C) 2004 by * +* Jason Kivlighn ([email protected]) * +* Unai Garro ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "recipefilter.h" + +#include <kdebug.h> + +#include "widgets/recipelistview.h" + +RecipeFilter::RecipeFilter( TDEListView *klv ) : TQObject( klv ), + listview( klv ), + currentCategory( 0 ) +{} + +void RecipeFilter::filter( const TQString &s ) +{ + //do this to only iterate over children of 'currentCategory' + TQListViewItem * pEndItem = NULL; + if ( currentCategory ) { + TQListViewItem * pStartItem = currentCategory; + do { + if ( pStartItem->nextSibling() ) + pEndItem = pStartItem->nextSibling(); + else + pStartItem = pStartItem->parent(); + } + while ( pStartItem && !pEndItem ); + } + + //Re-show everything + TQListViewItemIterator list_it; + if ( currentCategory ) + list_it = TQListViewItemIterator( currentCategory ); + else + list_it = TQListViewItemIterator( listview ); + while ( list_it.current() != pEndItem ) { + list_it.current() ->setVisible( true ); + list_it++; + } + + // Only filter if the filter text isn't empty + if ( !s.isEmpty() ) { + TQListViewItemIterator list_it( listview ); + while ( TQListViewItem * it = list_it.current() ) { + if ( it->rtti() == 1000 ) // Its a recipe + { + RecipeListItem * recipe_it = ( RecipeListItem* ) it; + + if ( recipe_it->title().contains( s, false ) ) + { + if ( currentCategory ) { + if ( isParentOf( currentCategory, recipe_it ) ) + recipe_it->setVisible( true ); + else + recipe_it->setVisible( false ); + } + else + recipe_it->setVisible( true ); + } + else + recipe_it->setVisible( false ); + } + + ++list_it; + } + hideIfEmpty(); + } +} + +void RecipeFilter::filterCategory( int categoryID ) +{ + kdDebug() << "I got category :" << categoryID << "\n"; + + if ( categoryID == -1 ) + currentCategory = 0; + else { + TQListViewItemIterator list_it( listview ); + while ( TQListViewItem * it = list_it.current() ) { + if ( it->rtti() == 1001 ) { + CategoryListItem * cat_it = ( CategoryListItem* ) it; + if ( cat_it->categoryId() == categoryID ) { + currentCategory = cat_it; + break; + } + } + + ++list_it; + } + } + + TQListViewItemIterator list_it( listview ); + while ( TQListViewItem * it = list_it.current() ) { + if ( categoryID == -1 ) + it->setVisible( true ); // We're not filtering categories + else if ( it == currentCategory || isParentOf( it, currentCategory ) || isParentOf( currentCategory, it ) ) + it->setVisible( true ); + else + it->setVisible( false ); + + ++list_it; + } + + if ( currentCategory ) + currentCategory->setOpen( true ); +} + +bool RecipeFilter::hideIfEmpty( TQListViewItem *parent ) +{ + TQListViewItem * it; + if ( parent == 0 ) + it = listview->firstChild(); + else + it = parent->firstChild(); + + bool parent_should_show = false; + for ( ; it; it = it->nextSibling() ) { + if ( (it->rtti() == 1000 && it->isVisible()) || (it->rtti() == NEXTLISTITEM_RTTI || it->rtti() == PREVLISTITEM_RTTI) ) { + parent_should_show = true; + } + else { + bool result = hideIfEmpty( it ); + if ( parent_should_show == false ) + parent_should_show = result; + } + } + + if ( parent && parent->rtti() != 1000 ) { + if ( parent_should_show ) + parent->setOpen( true ); + parent->setVisible( parent_should_show ); + } + return parent_should_show; +} + +bool RecipeFilter::isParentOf( TQListViewItem *parent, TQListViewItem *to_check ) +{ + for ( TQListViewItem * it = to_check->parent(); it; it = it->parent() ) { + if ( it == parent ) + return true; + } + + return false; +} + +#include "recipefilter.moc" + diff --git a/src/recipefilter.h b/src/recipefilter.h new file mode 100644 index 0000000..1418bfd --- /dev/null +++ b/src/recipefilter.h @@ -0,0 +1,39 @@ +/*************************************************************************** +* Copyright (C) 2004 by * +* Jason Kivlighn ([email protected]) * +* Unai Garro ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef RECIPEFILTER_H +#define RECIPEFILTER_H + +#include <tqobject.h> + +class TDEListView; +class TQListViewItem; + +class RecipeFilter : public TQObject +{ + TQ_OBJECT + +public: + RecipeFilter( TDEListView * ); + +public slots: + void filter( const TQString & ); + void filterCategory( int categoryID ); + +private: + bool isParentOf( TQListViewItem *parent, TQListViewItem *to_check ); + bool hideIfEmpty( TQListViewItem *parent = 0 ); + + TDEListView *listview; + TQListViewItem *currentCategory; +}; + +#endif //RECIPEFILTER_H diff --git a/src/setupwizard.cpp b/src/setupwizard.cpp new file mode 100644 index 0000000..73921ca --- /dev/null +++ b/src/setupwizard.cpp @@ -0,0 +1,850 @@ +/*************************************************************************** +* Copyright (C) 2003-2004 by * +* Unai Garro ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifdef HAVE_CONFIG_H +#include "config.h" +#endif + +#include <unistd.h> +#include <pwd.h> + +#include <tqhbox.h> +#include <tqvgroupbox.h> +#include <tqlayout.h> +#include <tqpixmap.h> +#include <tqpushbutton.h> +#include <tqtooltip.h> + +#include <tdeconfig.h> +#include <kdebug.h> +#include <tdeapplication.h> +#include <kstandarddirs.h> +#include <tdelocale.h> +#include <kiconloader.h> +#include <tdefiledialog.h> + +#include "setupwizard.h" + +SetupWizard::SetupWizard( TQWidget *parent, const char *name, bool modal, WFlags f ) : KWizard( parent, name, modal, f ) +{ + welcomePage = new WelcomePage( this ); + addPage( welcomePage, i18n( "Welcome to Krecipes" ) ); + + dbTypeSetupPage = new DBTypeSetupPage( this ); + addPage( dbTypeSetupPage, i18n( "Database Type" ) ); + + sqliteSetupPage = new SQLiteSetupPage( this ); + addPage( sqliteSetupPage, i18n( "Server Settings" ) ); + + permissionsSetupPage = new PermissionsSetupPage( this ); + addPage( permissionsSetupPage, i18n( "Database Permissions" ) ); + + pSqlPermissionsSetupPage = new PSqlPermissionsSetupPage( this ); + addPage( pSqlPermissionsSetupPage, i18n( "Database Permissions" ) ); + + serverSetupPage = new ServerSetupPage( this ); + addPage( serverSetupPage, i18n( "Server Settings" ) ); + + dataInitializePage = new DataInitializePage( this ); + addPage( dataInitializePage, i18n( "Initialize Database" ) ); + + savePage = new SavePage( this ); + addPage( savePage, i18n( "Finish & Save Settings" ) ); + + setFinishEnabled( savePage, true ); // Enable finish button + setSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::MinimumExpanding ); + + #if (!(HAVE_SQLITE || HAVE_SQLITE3)) + #if (HAVE_MYSQL) + showPages( MySQL ); + #else + #if (HAVE_POSTGRESQL) + showPages( PostgreSQL ); + #endif + #endif + #else + showPages( SQLite ); + #endif + + connect( finishButton(), TQ_SIGNAL( clicked() ), this, TQ_SLOT( save() ) ); + connect( dbTypeSetupPage, TQ_SIGNAL( showPages( DBType ) ), this, TQ_SLOT( showPages( DBType ) ) ); + +} + + +SetupWizard::~SetupWizard() +{ +} + + +void SetupWizard::showPages( DBType type ) +{ + switch ( type ) { + case MySQL: + setAppropriate( serverSetupPage, true ); + setAppropriate( permissionsSetupPage, true ); + setAppropriate( pSqlPermissionsSetupPage, false ); + setAppropriate( sqliteSetupPage, false ); + break; + case PostgreSQL: + setAppropriate( serverSetupPage, true ); + setAppropriate( pSqlPermissionsSetupPage, true ); + setAppropriate( permissionsSetupPage, false ); + setAppropriate( sqliteSetupPage, false ); + break; + case SQLite: + setAppropriate( serverSetupPage, false ); + setAppropriate( permissionsSetupPage, false ); + setAppropriate( pSqlPermissionsSetupPage, false ); + setAppropriate( sqliteSetupPage, true ); + break; + } +} + + +WelcomePage::WelcomePage( TQWidget *parent ) : TQWidget( parent ) +{ + TQGridLayout * layout = new TQGridLayout( this, 1, 1, 0, 0 ); + TQSpacerItem *spacer_top = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( spacer_top, 0, 1 ); + TQSpacerItem *spacer_left = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::Minimum ); + layout->addItem( spacer_left, 1, 0 ); + TQPixmap logoPixmap ( locate( "data", "krecipes/pics/wizard.png" ) ); + logo = new TQLabel( this ); + logo->setPixmap( logoPixmap ); + logo->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + layout->addWidget( logo, 1, 1, TQt::AlignTop ); + + TQSpacerItem *spacer_from_image = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::Minimum ); + layout->addItem( spacer_from_image, 1, 2 ); + + welcomeText = new TQLabel( this ); + welcomeText->setText( i18n( "<b><font size=\"+1\">Thank you very much for choosing Krecipes.</font></b><br>It looks like this is the first time you are using it. This wizard will help you with the initial setup so that you can start using it quickly.<br><br>Welcome, and enjoy cooking!" ) ); + welcomeText->setMinimumWidth( 200 ); + welcomeText->setMaximumWidth( 10000 ); + welcomeText->setSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::Minimum ); + welcomeText->setAlignment( int( TQLabel::WordBreak | TQLabel::AlignTop ) ); + layout->addWidget( welcomeText, 1, 3 ); + +} + +PermissionsSetupPage::PermissionsSetupPage( TQWidget *parent ) : TQWidget( parent ) +{ + TQGridLayout * layout = new TQGridLayout( this, 1, 1, 0, 0 ); + TQSpacerItem *spacer_top = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( spacer_top, 0, 1 ); + TQSpacerItem *spacer_left = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::Minimum ); + layout->addItem( spacer_left, 1, 0 ); + + + // Logo + TQPixmap permissionsSetupPixmap ( locate( "data", "krecipes/pics/dbpermissions.png" ) ); + logo = new TQLabel( this ); + logo->setPixmap( permissionsSetupPixmap ); + logo->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + layout->addMultiCellWidget( logo, 1, 8, 1, 1, TQt::AlignTop ); + + // Spacer to separate the logo + TQSpacerItem *logoSpacer = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::Minimum ); + layout->addItem( logoSpacer, 1, 2 ); + + + // Explanation Text + permissionsText = new TQLabel( this ); + permissionsText->setText( i18n( "This dialog will allow you to specify a MySQL account that has the necessary permissions to access the Krecipes MySQL database.<br><br><b><font size=\"+1\">Most users that use Krecipes and MySQL for the first time can just leave the default parameters and press \'Next\'.</font></b> <br><br>If you set a MySQL root password before, or you have already permissions as normal user, click on the appropriate option. Otherwise the account 'root' will be used, with no password.<br><br>[For security reasons, we strongly encourage you to setup a MySQL root password if you have not done so yet. Just type as root: mysqladmin password <i>your_password</i>]" ) ); + + permissionsText->setSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::Minimum ); + permissionsText->setAlignment( int( TQLabel::WordBreak | TQLabel::AlignTop ) ); + layout->addWidget( permissionsText, 1, 3 ); + + // Text spacer + TQSpacerItem *textSpacer = new TQSpacerItem( 10, 30, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( textSpacer, 2, 3 ); + + + // "The user already has permissions" checkbox + noSetupCheckBox = new TQCheckBox( i18n( "I have already set the necessary permissions" ), this, "noSetupCheckBox" ); + layout->addWidget( noSetupCheckBox, 3, 3 ); + + TQSpacerItem *checkBoxSpacer = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( checkBoxSpacer, 4, 3 ); + + // root checkbox + rootCheckBox = new TQCheckBox( i18n( "I have already set a MySQL root/admin account" ), this, "rootCheckBox" ); + layout->addWidget( rootCheckBox, 5, 3 ); + + TQSpacerItem *rootInfoSpacer = new TQSpacerItem( 10, 20, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( rootInfoSpacer, 6, 3 ); + + // MySQL root/admin info + TQGroupBox *rootInfoGBox = new TQGroupBox( this, "rootInfoGBox" ); + rootInfoGBox->setTitle( i18n( "MySQL Administrator Account" ) ); + rootInfoGBox->setEnabled( false ); // Disable by default + rootInfoGBox->setColumns( 2 ); + rootInfoGBox->setInsideSpacing( 10 ); + layout->addWidget( rootInfoGBox, 7, 3 ); + + // User Entry + TQLabel *userLabel = new TQLabel( rootInfoGBox ); + userLabel->setText( i18n( "Username:" ) ); + userEdit = new KLineEdit( rootInfoGBox ); + userEdit->setText( "root" ); + + // Password Entry + TQLabel *passLabel = new TQLabel( rootInfoGBox ); + passLabel->setText( i18n( "Password:" ) ); + passEdit = new KLineEdit( rootInfoGBox ); + passEdit->setEchoMode( TQLineEdit::Password ); + + // Bottom spacer + TQSpacerItem *bottomSpacer = new TQSpacerItem( 10, 20, TQSizePolicy::Minimum, TQSizePolicy::MinimumExpanding ); + layout->addItem( bottomSpacer, 8, 1 ); + + // Connect Signals & slots + + connect( rootCheckBox, TQ_SIGNAL( toggled( bool ) ), rootInfoGBox, TQ_SLOT( setEnabled( bool ) ) ); + connect( rootCheckBox, TQ_SIGNAL( toggled( bool ) ), this, TQ_SLOT( rootCheckBoxChanged( bool ) ) ); + connect( noSetupCheckBox, TQ_SIGNAL( toggled( bool ) ), this, TQ_SLOT( noSetupCheckBoxChanged( bool ) ) ); +} + +void PermissionsSetupPage::rootCheckBoxChanged( bool on ) +{ + if ( on ) + noSetupCheckBox->setChecked( false ); // exclude mutually the options (both can be unset) +} + +bool PermissionsSetupPage::doUserSetup() +{ + return ( !noSetupCheckBox->isChecked() ); +} + +bool PermissionsSetupPage::useAdmin() +{ + return ( rootCheckBox->isChecked() ); +} + +void PermissionsSetupPage::getAdmin( TQString &adminName, TQString &adminPass ) +{ + adminName = userEdit->text(); + adminPass = passEdit->text(); +} + +void PermissionsSetupPage::noSetupCheckBoxChanged( bool on ) +{ + if ( on ) + rootCheckBox->setChecked( false ); // exclude mutually the options (both can be unset) +} + + +PSqlPermissionsSetupPage::PSqlPermissionsSetupPage( TQWidget *parent ) : TQWidget( parent ) +{ + TQGridLayout * layout = new TQGridLayout( this, 1, 1, 0, 0 ); + TQSpacerItem *spacer_top = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( spacer_top, 0, 1 ); + TQSpacerItem *spacer_left = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::Minimum ); + layout->addItem( spacer_left, 1, 0 ); + + + // Logo + TQPixmap permissionsSetupPixmap ( locate( "data", "krecipes/pics/dbpermissions.png" ) ); + logo = new TQLabel( this ); + logo->setPixmap( permissionsSetupPixmap ); + logo->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + layout->addMultiCellWidget( logo, 1, 8, 1, 1, TQt::AlignTop ); + + // Spacer to separate the logo + TQSpacerItem *logoSpacer = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::Minimum ); + layout->addItem( logoSpacer, 1, 2 ); + + + // Explanation Text + permissionsText = new TQLabel( this ); + permissionsText->setText( i18n( "This dialog will allow you to specify a PostgreSQL account that has the necessary permissions to access the Krecipes PostgreSQL database. This account may either be a <b>PostgreSQL superuser</b> or have the ability to both <b>create new PostgreSQL users and databases</b>.<br><br>If no superuser or privileged account is given, the account 'postgres' will be attempted, with no password. If this is insufficient for your PostgreSQL setup, you <b>must</b> select the appropriate option below to enter the information of a privileged PostgreSQL account." ) ); + + permissionsText->setSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::Minimum ); + permissionsText->setAlignment( int( TQLabel::WordBreak | TQLabel::AlignTop ) ); + layout->addWidget( permissionsText, 1, 3 ); + + // Text spacer + TQSpacerItem *textSpacer = new TQSpacerItem( 10, 30, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( textSpacer, 2, 3 ); + + + // "The user already has permissions" checkbox + noSetupCheckBox = new TQCheckBox( i18n( "I have already set the necessary permissions" ), this, "noSetupCheckBox" ); + layout->addWidget( noSetupCheckBox, 3, 3 ); + + TQSpacerItem *checkBoxSpacer = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( checkBoxSpacer, 4, 3 ); + + // root checkbox + rootCheckBox = new TQCheckBox( i18n( "I have already set a superuser or privileged account" ), this, "rootCheckBox" ); + layout->addWidget( rootCheckBox, 5, 3 ); + + TQSpacerItem *rootInfoSpacer = new TQSpacerItem( 10, 20, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( rootInfoSpacer, 6, 3 ); + + // MySQL root/admin info + TQGroupBox *rootInfoGBox = new TQGroupBox( this, "rootInfoGBox" ); + rootInfoGBox->setTitle( i18n( "PostgreSQL Superuser or Privileged Account" ) ); + rootInfoGBox->setEnabled( false ); // Disable by default + rootInfoGBox->setColumns( 2 ); + rootInfoGBox->setInsideSpacing( 10 ); + layout->addWidget( rootInfoGBox, 7, 3 ); + + // User Entry + TQLabel *userLabel = new TQLabel( rootInfoGBox ); + userLabel->setText( i18n( "Username:" ) ); + userEdit = new KLineEdit( rootInfoGBox ); + userEdit->setText( "postgres" ); + + // Password Entry + TQLabel *passLabel = new TQLabel( rootInfoGBox ); + passLabel->setText( i18n( "Password:" ) ); + passEdit = new KLineEdit( rootInfoGBox ); + passEdit->setEchoMode( TQLineEdit::Password ); + + // Bottom spacer + TQSpacerItem *bottomSpacer = new TQSpacerItem( 10, 20, TQSizePolicy::Minimum, TQSizePolicy::MinimumExpanding ); + layout->addItem( bottomSpacer, 8, 1 ); + + // Connect Signals & slots + + connect( rootCheckBox, TQ_SIGNAL( toggled( bool ) ), rootInfoGBox, TQ_SLOT( setEnabled( bool ) ) ); + connect( rootCheckBox, TQ_SIGNAL( toggled( bool ) ), this, TQ_SLOT( rootCheckBoxChanged( bool ) ) ); + connect( noSetupCheckBox, TQ_SIGNAL( toggled( bool ) ), this, TQ_SLOT( noSetupCheckBoxChanged( bool ) ) ); +} + +void PSqlPermissionsSetupPage::rootCheckBoxChanged( bool on ) +{ + if ( on ) + noSetupCheckBox->setChecked( false ); // exclude mutually the options (both can be unset) +} + +bool PSqlPermissionsSetupPage::doUserSetup() +{ + return ( !noSetupCheckBox->isChecked() ); +} + +bool PSqlPermissionsSetupPage::useAdmin() +{ + return ( rootCheckBox->isChecked() ); +} + +void PSqlPermissionsSetupPage::getAdmin( TQString &adminName, TQString &adminPass ) +{ + adminName = userEdit->text(); + adminPass = passEdit->text(); +} + +void PSqlPermissionsSetupPage::noSetupCheckBoxChanged( bool on ) +{ + if ( on ) + rootCheckBox->setChecked( false ); // exclude mutually the options (both can be unset) +} + + +ServerSetupPage::ServerSetupPage( TQWidget *parent ) : TQWidget( parent ) +{ + TQGridLayout * layout = new TQGridLayout( this, 1, 1, 0, 0 ); + TQSpacerItem *spacer_top = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( spacer_top, 0, 1 ); + TQSpacerItem *spacer_left = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::Minimum ); + layout->addItem( spacer_left, 1, 0 ); + + + // Image + + TQPixmap serverSetupPixmap ( locate( "data", "krecipes/pics/network.png" ) ); + logo = new TQLabel( this ); + logo->setPixmap( serverSetupPixmap ); + logo->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + layout->addMultiCellWidget( logo, 1, 8, 1, 1, TQt::AlignTop ); + + TQSpacerItem *spacer_from_image = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::Minimum ); + layout->addItem( spacer_from_image, 1, 2 ); + + + // Explanation text + serverSetupText = new TQLabel( this ); + serverSetupText->setText( i18n( "In this dialog you can adjust the database server settings.<br><br><b>Warning: Passwords are stored in plain text and could potentially be compromised. We recommend that you create a username and password combination solely for use by Krecipes.</b>" ) ); + serverSetupText->setSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::Minimum ); + serverSetupText->setAlignment( int( TQLabel::AlignTop | TQLabel::WordBreak ) ); + layout->addWidget( serverSetupText, 1, 3 ); + + // Text spacer + + TQSpacerItem* textSpacer = new TQSpacerItem( 10, 30, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( textSpacer, 2, 3 ); + + // Input Boxes + + TQGroupBox *inputGBox = new TQGroupBox( this, "inputGBox" ); + inputGBox->setFrameStyle( TQFrame::NoFrame ); + inputGBox->setInsideSpacing( 10 ); + inputGBox->setColumns( 2 ); + layout->addWidget( inputGBox, 3, 3 ); + + // Username Input + + TQLabel* usernameText = new TQLabel( i18n( "Username:" ), inputGBox ); + usernameText->setFixedSize( TQSize( 100, 20 ) ); + usernameText->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + + usernameEdit = new KLineEdit( inputGBox ); + usernameEdit->setFixedSize( TQSize( 120, 20 ) ); + usernameEdit->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + // get username + uid_t userID; + TQString username; + struct passwd *user; + userID = getuid(); + user = getpwuid ( userID ); + username = user->pw_name; + + usernameEdit->setText( username ); + + + // Password + + TQLabel* passwordText = new TQLabel( i18n( "Password:" ), inputGBox ); + passwordText->setFixedSize( TQSize( 100, 20 ) ); + passwordText->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + + passwordEdit = new KLineEdit( inputGBox ); + passwordEdit->setEchoMode( TQLineEdit::Password ); + passwordEdit->setFixedSize( TQSize( 120, 20 ) ); + passwordEdit->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + + // DB Name + + TQLabel* dbNameText = new TQLabel( i18n( "Database name:" ), inputGBox ); + dbNameText->setFixedSize( TQSize( 100, 20 ) ); + dbNameText->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + + dbNameEdit = new KLineEdit( inputGBox ); + dbNameEdit->setFixedSize( TQSize( 120, 20 ) ); + dbNameEdit->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + dbNameEdit->setText( "Krecipes" ); + + + // Spacer from box + TQSpacerItem* spacerFromBox = new TQSpacerItem( 10, 20, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( spacerFromBox, 4, 3 ); + + + // Remote server checkbox + + remoteServerCheckBox = new TQCheckBox( i18n( "The server is remote" ), this, "remoteServerCheckBox" ); + layout->addWidget( remoteServerCheckBox, 5, 3 ); + + // Spacer from CheckBox + TQSpacerItem* spacerFromCheckBox = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( spacerFromCheckBox, 6, 3 ); + + // Server & Client Box + TQGroupBox *serverSettingsGBox = new TQGroupBox( this, "serverSettingsGBox" ); + serverSettingsGBox->setTitle( i18n( "Server / Client Settings" ) ); + serverSettingsGBox->setEnabled( false ); // Disable by default + serverSettingsGBox->setInsideSpacing( 10 ); + serverSettingsGBox->setColumns( 2 ); + layout->addWidget( serverSettingsGBox, 7, 3 ); + + + // Server + ( void ) new TQLabel( i18n( "Server:" ), serverSettingsGBox ); + serverEdit = new KLineEdit( serverSettingsGBox ); + serverEdit->setText( "localhost" ); + + // Client + ( void ) new TQLabel( i18n( "Client:" ), serverSettingsGBox ); + clientEdit = new KLineEdit( serverSettingsGBox ); + clientEdit->setText( "localhost" ); + + // Bottom Spacers + + TQSpacerItem* bottomSpacer = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::MinimumExpanding ); + layout->addItem( bottomSpacer, 8, 1 ); + + //TQSpacerItem* spacerRight = new TQSpacerItem( 10, 10, TQSizePolicy::MinimumExpanding, TQSizePolicy::Fixed ); + //layout->addItem( spacerRight, 7, 4 ); + + // Signals & Slots + connect( remoteServerCheckBox, TQ_SIGNAL( toggled( bool ) ), serverSettingsGBox, TQ_SLOT( setEnabled( bool ) ) ); + +} + +TQString ServerSetupPage::server( void ) +{ + return ( serverEdit->text() ); +} + +TQString ServerSetupPage::user( void ) +{ + return ( usernameEdit->text() ); +} + +TQString ServerSetupPage::password( void ) +{ + return ( passwordEdit->text() ); +} + +TQString ServerSetupPage::dbName( void ) +{ + return ( dbNameEdit->text() ); +} + +void ServerSetupPage::getServerInfo( bool &isRemote, TQString &host, TQString &client, TQString &dbName, TQString &user, TQString &pass, int &port ) +{ + isRemote = remoteServerCheckBox->isChecked(); + host = serverEdit->text(); + client = clientEdit->text(); + user = usernameEdit->text(); + pass = passwordEdit->text(); + dbName = dbNameEdit->text(); + port = 0; +} + +SQLiteSetupPage::SQLiteSetupPage( TQWidget *parent ) : TQWidget( parent ) +{ + TQGridLayout * layout = new TQGridLayout( this, 1, 1, 0, 0 ); + TQSpacerItem *spacer_top = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( spacer_top, 0, 1 ); + TQSpacerItem *spacer_left = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::Minimum ); + layout->addItem( spacer_left, 1, 0 ); + + + // Image + + TQPixmap serverSetupPixmap ( locate( "data", "krecipes/pics/network.png" ) ); + logo = new TQLabel( this ); + logo->setPixmap( serverSetupPixmap ); + logo->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + layout->addMultiCellWidget( logo, 1, 4, 1, 1, TQt::AlignTop ); + + TQSpacerItem *spacer_from_image = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::Minimum ); + layout->addItem( spacer_from_image, 1, 2 ); + + + // Explanation text + serverSetupText = new TQLabel( this ); + serverSetupText->setText( i18n( "In this dialog you can adjust SQLite settings." ) ); + serverSetupText->setSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::Minimum ); + serverSetupText->setAlignment( int( TQLabel::AlignTop | TQLabel::AlignJustify ) ); + layout->addWidget( serverSetupText, 1, 3 ); + + // Text spacer + + TQSpacerItem* textSpacer = new TQSpacerItem( 10, 30, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( textSpacer, 2, 3 ); + + // Input Boxes + + TQHBox *hbox = new TQHBox( this ); + + ( void ) new TQLabel( i18n( "Database file:" ), hbox ); + + fileEdit = new KLineEdit( hbox ); + fileEdit->setText( locateLocal ( "appdata", "krecipes.krecdb" ) ); + hbox->setStretchFactor( fileEdit, 2 ); + + TDEIconLoader il; + TQPushButton *file_select = new TQPushButton( il.loadIcon( "document-open", TDEIcon::NoGroup, 16 ), TQString::null, hbox ); + TQToolTip::add + ( file_select, i18n( "Open file dialog" ) ); + file_select->setFixedWidth( 25 ); + + layout->addWidget( hbox, 3, 3 ); + + // Bottom Spacers + + TQSpacerItem* bottomSpacer = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::MinimumExpanding ); + layout->addItem( bottomSpacer, 4, 1 ); + + connect( file_select, TQ_SIGNAL( clicked() ), TQ_SLOT( selectFile() ) ); +} + +TQString SQLiteSetupPage::dbFile( void ) const +{ + return ( fileEdit->text() ); +} + +void SQLiteSetupPage::selectFile() +{ + KFileDialog dialog( TQString::null, "*.*|All Files", this, "dialog", true ); + if ( dialog.exec() == TQDialog::Accepted ) { + fileEdit->setText( dialog.selectedFile() ); + } +} + + +SavePage::SavePage( TQWidget *parent ) : TQWidget( parent ) +{ + TQGridLayout * layout = new TQGridLayout( this, 1, 1, 0, 0 ); + TQSpacerItem *spacer_top = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( spacer_top, 0, 1 ); + TQSpacerItem *spacer_left = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::Minimum ); + layout->addItem( spacer_left, 1, 0 ); + + TQPixmap logoPixmap ( locate( "data", "krecipes/pics/save.png" ) ); + logo = new TQLabel( this ); + logo->setPixmap( logoPixmap ); + logo->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + layout->addWidget( logo, 1, 1, TQt::AlignTop ); + + TQSpacerItem *spacer_from_image = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::Minimum ); + layout->addItem( spacer_from_image, 1, 2 ); + + saveText = new TQLabel( this ); + saveText->setText( i18n( "Congratulations; all the necessary configuration setup is done. Press 'Finish' to continue, and enjoy cooking!" ) ); + saveText->setSizePolicy( TQSizePolicy::Minimum, TQSizePolicy::Minimum ); + + saveText->setAlignment( int( TQLabel::WordBreak | TQLabel::AlignVCenter ) ); + layout->addWidget( saveText, 1, 3 ); + + +} + +void SetupWizard::save( void ) +{ + kdDebug() << "Setting parameters in tdeconfig..." << endl; + TDEConfig *config = kapp->config(); + + // Save the database type + TQString sDBType; + + switch ( dbTypeSetupPage->dbType() ) { + case MySQL: + sDBType = "MySQL"; + break; + case PostgreSQL: + sDBType = "PostgreSQL"; + break; + default: + sDBType = "SQLite"; + break; + } + + config->setGroup( "DBType" ); + config->writeEntry( "Type", sDBType ); + kdDebug() << "DB type set in tdeconfig was... " << sDBType << endl; + // Save the server data if needed + if ( !( dbTypeSetupPage->dbType() == SQLite ) ) { + config->setGroup( "Server" ); + config->writeEntry( "Host", serverSetupPage->server() ); + config->writeEntry( "Username", serverSetupPage->user() ); + config->writeEntry( "Password", serverSetupPage->password() ); + config->writeEntry( "DBName", serverSetupPage->dbName() ); + kdDebug() << "Finished setting the database parameters for MySQL or PostgreSQL (non SQLite)..." << endl; + } + else { + config->setGroup( "Server" ); + config->writeEntry( "DBFile", sqliteSetupPage->dbFile() ); + } + + // Indicate that settings were already made + + config->setGroup( "Wizard" ); + config->writeEntry( "SystemSetup", true ); + config->writeEntry( "Version", "0.9" ); + kdDebug() << "Setting in tdeconfig the lines to disable wizard startup..." << sDBType << endl; +} + +void SetupWizard::getOptions( bool &setupUser, bool &initializeData, bool &doUSDAImport ) +{ + setupUser = permissionsSetupPage->doUserSetup() && pSqlPermissionsSetupPage->doUserSetup(); + initializeData = dataInitializePage->doInitialization(); + doUSDAImport = dataInitializePage->doUSDAImport(); +} + +void SetupWizard::getAdminInfo( bool &enabled, TQString &adminUser, TQString &adminPass, const TQString &dbType ) +{ + enabled = permissionsSetupPage->useAdmin() || pSqlPermissionsSetupPage->useAdmin(); + if ( dbType == "MySQL" ) + permissionsSetupPage->getAdmin( adminUser, adminPass ); + else + pSqlPermissionsSetupPage->getAdmin( adminUser, adminPass ); +} + +void SetupWizard::getServerInfo( bool &isRemote, TQString &host, TQString &client, TQString &dbName, TQString &user, TQString &pass, int &port ) +{ + serverSetupPage->getServerInfo( isRemote, host, client, dbName, user, pass, port ); + if ( dbTypeSetupPage->dbType() == SQLite ) + dbName = sqliteSetupPage->dbFile(); +} + +DataInitializePage::DataInitializePage( TQWidget *parent ) : TQWidget( parent ) +{ + TQGridLayout * layout = new TQGridLayout( this, 1, 1, 0, 0 ); + TQSpacerItem *spacer_top = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( spacer_top, 0, 1 ); + TQSpacerItem *spacer_left = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::Minimum ); + layout->addItem( spacer_left, 1, 0 ); + + // Explanation Text// Widgets + TQLabel *logo; + //TQLabel *serverSetupText; + //KLineEdit *usernameEdit; + //KLineEdit *passwordEdit; + //KLineEdit *dbNameEdit; + initializeText = new TQLabel( this ); + initializeText->setText( i18n( "Krecipes comes with some delicious default recipes and useful data. <br><br>Would you like to initialize your database with those? Note that this will erase all your previous recipes if you have any. " ) ); + + initializeText->setSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::Minimum ); + initializeText->setAlignment( int( TQLabel::WordBreak | TQLabel::AlignTop ) ); + layout->addWidget( initializeText, 1, 3 ); + + // Logo + TQPixmap dataInitializePixmap ( locate( "data", "krecipes/pics/recipes.png" ) ); + logo = new TQLabel( this ); + logo->setPixmap( dataInitializePixmap ); + logo->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + layout->addMultiCellWidget( logo, 1, 8, 1, 1, TQt::AlignTop ); + + // Spacer to separate the logo + TQSpacerItem *logoSpacer = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::Minimum ); + layout->addItem( logoSpacer, 1, 2 ); + + // Initialize data checkbox + + initializeCheckBox = new TQCheckBox( i18n( "Yes please, initialize the database with the examples" ), this, "initializeCheckBox" ); + layout->addWidget( initializeCheckBox, 3, 3 ); + + TQSpacerItem *textInfoSpacer = new TQSpacerItem( 0, 50, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( textInfoSpacer, 4, 3 ); + + USDAImportText = new TQLabel( i18n( "Krecipes can import nutrient data from the USDA's nutrient database for over 400 foods. A total of 43 food properties are included for each food, such as energy, fat, vitamin C, etc.<br><br>Would you like to import this data now? Note that this operation is safe to use on an existing database, and no data loss will occur. This operation may take several minutes." ), this ); + layout->addWidget( USDAImportText, 5, 3 ); + + TQSpacerItem *importInfoSpacer = new TQSpacerItem( 0, 50, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( importInfoSpacer, 6, 3 ); + + USDAImportCheckBox = new TQCheckBox( i18n( "Yes please, load the database with nutrient data for 400+ foods. (Note: English only.)" ), this, "USDAImportCheckBox" ); + layout->addWidget( USDAImportCheckBox, 7, 3 ); +} + +bool DataInitializePage::doInitialization( void ) +{ + return ( initializeCheckBox->isChecked() ); +} + +bool DataInitializePage::doUSDAImport( void ) +{ + return ( USDAImportCheckBox->isChecked() ); +} + +DBTypeSetupPage::DBTypeSetupPage( TQWidget *parent ) : TQWidget( parent ) +{ + TQGridLayout * layout = new TQGridLayout( this, 1, 1, 0, 0 ); + TQSpacerItem *spacer_top = new TQSpacerItem( 10, 10, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( spacer_top, 0, 1 ); + TQSpacerItem *spacer_left = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::Minimum ); + layout->addItem( spacer_left, 1, 0 ); + + + // Image + + TQPixmap serverSetupPixmap ( locate( "data", "krecipes/pics/network.png" ) ); + logo = new TQLabel( this ); + logo->setPixmap( serverSetupPixmap ); + logo->setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Fixed ); + layout->addMultiCellWidget( logo, 1, 4, 1, 1, TQt::AlignTop ); + + TQSpacerItem *spacer_from_image = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::Minimum ); + layout->addItem( spacer_from_image, 1, 2 ); + + + // Explanation text + dbTypeSetupText = new TQLabel( this ); + dbTypeSetupText->setText( i18n( "Choose the type of database that you want to use. Most users will want to choose a simple local database here. However, you can also use remote servers by means of a MySQL or PostgreSQL database." ) ); + dbTypeSetupText->setSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::Minimum ); + dbTypeSetupText->setAlignment( int( TQLabel::AlignTop | TQLabel::WordBreak ) ); + layout->addWidget( dbTypeSetupText, 1, 3 ); + + // Text spacer + + TQSpacerItem* textSpacer = new TQSpacerItem( 10, 30, TQSizePolicy::Minimum, TQSizePolicy::Fixed ); + layout->addItem( textSpacer, 2, 3 ); + + + + // Database type choice + bg = new TQVButtonGroup( this ); + layout->addWidget( bg, 3, 3 ); + + liteCheckBox = new TQRadioButton( i18n( "Simple Local File (SQLite)" ), bg, "liteCheckBox" ); + mysqlCheckBox = new TQRadioButton( i18n( "Local or Remote MySQL Database" ), bg, "liteCheckBox" ); + psqlCheckBox = new TQRadioButton( i18n( "Local or Remote PostgreSQL Database" ), bg, "psqlCheckBox" ); + bg->setButton( 0 ); // By default, SQLite + +#if (!HAVE_MYSQL) + mysqlCheckBox->setEnabled( false ); +#endif + +#if (!HAVE_POSTGRESQL) + psqlCheckBox->setEnabled( false ); +#endif + +#if (!(HAVE_SQLITE || HAVE_SQLITE3)) + liteCheckBox->setEnabled( false ); +#if (HAVE_MYSQL) + + bg->setButton( 1 ); // Otherwise by default liteCheckBox is checked even if it's disabled +#else + #if (HAVE_POSTGRESQL) + + bg->setButton( 2 ); +#endif + #endif +#endif + + + TQSpacerItem *spacer_bottom = new TQSpacerItem( 10, 10, TQSizePolicy::Fixed, TQSizePolicy::MinimumExpanding ); + layout->addItem( spacer_bottom, 4, 3 ); + + connect( bg, TQ_SIGNAL( clicked( int ) ), this, TQ_SLOT( setPages( int ) ) ); + +} + +int DBTypeSetupPage::dbType( void ) +{ + int id = bg->id( bg->selected() ); + + switch ( id ) { + case 1: + return ( MySQL ); // MySQL (note index=0,1....) + case 2: + return ( PostgreSQL ); + default: + return ( SQLite ); + } +} + +/* +** hides/shows pages given the radio button state +*/ + +void DBTypeSetupPage::setPages( int rb ) +{ + switch ( rb ) { + case 1: + emit showPages( MySQL ); + break; + case 2: + emit showPages( PostgreSQL ); + break; + default: + emit showPages( SQLite ); + break; + } +} + +#include "setupwizard.moc" diff --git a/src/setupwizard.h b/src/setupwizard.h new file mode 100644 index 0000000..a3ceac3 --- /dev/null +++ b/src/setupwizard.h @@ -0,0 +1,224 @@ +/*************************************************************************** +* Copyright (C) 2003-2004 by * +* Unai Garro ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ +#ifndef SETUPWIZARD_H +#define SETUPWIZARD_H + +#include <tqvbuttongroup.h> +#include <tqcheckbox.h> +#include <tqradiobutton.h> +#include <tqlabel.h> +#include <kwizard.h> +#include <klineedit.h> +#include <ktextedit.h> + + +/** +@author Unai Garro +*/ + +class WelcomePage; +class DBTypeSetupPage; +class PermissionsSetupPage; +class PSqlPermissionsSetupPage; +class ServerSetupPage; +class DataInitializePage; +class SavePage; +class SQLiteSetupPage; + +enum DBType {SQLite, MySQL, PostgreSQL}; + +class SetupWizard: public KWizard +{ + TQ_OBJECT +public: + + SetupWizard( TQWidget *parent = 0, const char *name = 0, bool modal = true, WFlags f = 0 ); + ~SetupWizard(); + void getOptions( bool &setupUser, bool &initializeData, bool &doUSDAImport ); + void getAdminInfo( bool &enabled, TQString &adminUser, TQString &adminPass, const TQString &dbType ); + void getServerInfo( bool &isRemote, TQString &host, TQString &client, TQString &dbName, TQString &user, TQString &pass, int &port ); +private: + // Widgets + WelcomePage *welcomePage; + DBTypeSetupPage *dbTypeSetupPage; + PermissionsSetupPage *permissionsSetupPage; + PSqlPermissionsSetupPage *pSqlPermissionsSetupPage; + ServerSetupPage *serverSetupPage; + DataInitializePage *dataInitializePage; + SQLiteSetupPage *sqliteSetupPage; + SavePage *savePage; + +private slots: + void save( void ); + void showPages( DBType ); + +}; + +class WelcomePage: public TQWidget +{ +public: + // Methods + WelcomePage( TQWidget *parent ); + +private: + // Widgets + TQLabel *logo; + TQLabel *welcomeText; + +}; + + +class PermissionsSetupPage: public TQWidget +{ + TQ_OBJECT +public: + // Methods + PermissionsSetupPage( TQWidget *parent ); + bool doUserSetup( void ); + bool useAdmin( void ); + void getAdmin( TQString &adminName, TQString &adminPass ); +private: + // Widgets + TQLabel *logo; + TQLabel *permissionsText; + TQCheckBox *noSetupCheckBox; + TQCheckBox *rootCheckBox; + TQLineEdit *userEdit; + TQLineEdit *passEdit; + +private slots: + void rootCheckBoxChanged( bool on ); + void noSetupCheckBoxChanged( bool on ); + +}; + +class PSqlPermissionsSetupPage: public TQWidget +{ + TQ_OBJECT +public: + // Methods + PSqlPermissionsSetupPage( TQWidget *parent ); + bool doUserSetup( void ); + bool useAdmin( void ); + void getAdmin( TQString &adminName, TQString &adminPass ); +private: + // Widgets + TQLabel *logo; + TQLabel *permissionsText; + TQCheckBox *noSetupCheckBox; + TQCheckBox *rootCheckBox; + TQLineEdit *userEdit; + TQLineEdit *passEdit; + +private slots: + void rootCheckBoxChanged( bool on ); + void noSetupCheckBoxChanged( bool on ); + +}; + +class ServerSetupPage: public TQWidget +{ +public: + // Methods + ServerSetupPage( TQWidget *parent ); + TQString user( void ); + TQString password( void ); + TQString dbName( void ); + TQString server( void ); + TQString client( void ); + void getServerInfo( bool &isRemote, TQString &host, TQString &client, TQString &dbName, TQString &user, TQString &pass, int &port ); +private: + // Widgets + TQLabel *logo; + TQLabel *serverSetupText; + KLineEdit *usernameEdit; + KLineEdit *passwordEdit; + KLineEdit *dbNameEdit; + TQCheckBox *remoteServerCheckBox; + KLineEdit *serverEdit; + KLineEdit *clientEdit; +}; + + +class SQLiteSetupPage: public TQWidget +{ + TQ_OBJECT + +public: + // Methods + SQLiteSetupPage( TQWidget *parent ); + TQString dbFile() const; + +private slots: + void selectFile(); + +private: + // Widgets + TQLabel *logo; + TQLabel *serverSetupText; + KLineEdit *fileEdit; +}; + + +class DataInitializePage: public TQWidget +{ +public: + // Methods + DataInitializePage( TQWidget *parent ); + bool doInitialization( void ); + bool doUSDAImport( void ); + +private: + // Widgets + TQLabel *logo; + TQLabel *initializeText; + TQLabel *USDAImportText; + TQCheckBox *initializeCheckBox; + TQCheckBox *USDAImportCheckBox; + +}; + +class SavePage: public TQWidget +{ +public: + // Methods + SavePage( TQWidget *parent ); +private: + // Widgets + TQLabel *logo; + TQLabel *saveText; + +}; + +class DBTypeSetupPage: public TQWidget +{ + + TQ_OBJECT + +public: + // Methods + DBTypeSetupPage( TQWidget *parent ); + int dbType( void ); +private: + // Widgets + TQLabel *dbTypeSetupText; + TQLabel *logo; + TQVButtonGroup *bg; + TQRadioButton *liteCheckBox; + TQRadioButton *mysqlCheckBox; + TQRadioButton *psqlCheckBox; +private slots: + void setPages( int rb ); // hides/shows pages given the radio button state +signals: + void showPages( DBType ); +}; + +#endif diff --git a/src/shoppingcalculator.cpp b/src/shoppingcalculator.cpp new file mode 100644 index 0000000..e95624a --- /dev/null +++ b/src/shoppingcalculator.cpp @@ -0,0 +1,78 @@ +/*************************************************************************** +* Copyright (C) 2003 by Unai Garro * +* [email protected] * +* * +* 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. * +***************************************************************************/ +#include "shoppingcalculator.h" + +#include <kdebug.h> + +#include "propertycalculator.h" +#include "datablocks/elementlist.h" +#include "datablocks/ingredientlist.h" + +#include "backends/recipedb.h" + +//NOTE: The code as-is uses the lower value if an ingredient range exists +// However, the shopping list is calculated using the upper value +// because the values are already adjusted prior to being passed +// to these functions + +void calculateShopping( const ElementList &recipeList, IngredientList *ingredientList, RecipeDB *db ) +{ + for ( ElementList::const_iterator recipe_it = recipeList.begin(); recipe_it != recipeList.end(); ++recipe_it ) { + Recipe rec; + db->loadRecipe( &rec, RecipeDB::Ingredients, ( *recipe_it ).id ); + sum( ingredientList, &( rec.ingList ), db ); + } +} + +void sum( IngredientList *totalIngredientList, IngredientList *newIngredientList, RecipeDB *db ) +{ + for ( IngredientList::const_iterator ing_it = newIngredientList->begin(); ing_it != newIngredientList->end(); ++ing_it ) { + IngredientList::iterator pos_it; + + // Find out if ingredient exists in list already + int pos = totalIngredientList->find( ( *ing_it ).ingredientID ); + + if ( pos >= 0 ) // the ingredient is already listed + { + pos_it = totalIngredientList->at( pos ); + + // Variables to store the new total + Ingredient result; + bool converted; + + // Do the conversion + // try to with this and next in the list until conversion rate is + // found or end of list is reached + IngredientList::iterator lastpos_it; // for 'backup' + do + { + lastpos_it = pos_it; + + // Try to convert + converted = autoConvert( db, *ing_it, *pos_it, result ); + } + while ( ( !converted ) && ( ( ( pos_it = totalIngredientList->find( ++pos_it, ( *ing_it ).ingredientID ) ) ) != totalIngredientList->end() ) ); + + // If the conversion was succesful, Set the New Values + if ( converted ) { + *lastpos_it = result; + } + else // Otherwise append this ingredient at the end of the list + { + // Insert ingredient ID in the list + totalIngredientList->append( *ing_it ); + } + } + else // The ingredient is not in the list, just append + { + totalIngredientList->append( *ing_it ); + } + } +} diff --git a/src/shoppingcalculator.h b/src/shoppingcalculator.h new file mode 100644 index 0000000..1d00323 --- /dev/null +++ b/src/shoppingcalculator.h @@ -0,0 +1,23 @@ +/*************************************************************************** +* Copyright (C) 2003 by Unai Garro * +* [email protected] * +* * +* 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. * +***************************************************************************/ +#ifndef SHOPPINGCALCULATOR_H +#define SHOPPINGCALCULATOR_H + +class ElementList; +class IngredientList; +class RecipeDB; + +/** +@author Unai Garro +*/ +void calculateShopping( const ElementList &recipeList, IngredientList *ingredientList, RecipeDB *db ); +void sum ( IngredientList *totalIngredientList, IngredientList *newIngredientList, RecipeDB *db ); +int autoConvertUnits( RecipeDB* database, double amount1, int unit1, double amount2, int unit2, double &newAmount, int &newID ); +#endif diff --git a/src/tests/Makefile.am b/src/tests/Makefile.am new file mode 100644 index 0000000..2a5e97b --- /dev/null +++ b/src/tests/Makefile.am @@ -0,0 +1,19 @@ +INCLUDES = -I$(srcdir)/.. -I$(srcdir)/../importers -I$(srcdir)/../exporters $(all_includes) + +AM_LDFLAGS = $(KDE_RPATH) $(all_libraries) + +check_PROGRAMS = kretest mmftest mx2test mxptest rezkonvtest nyctest recipemltest + +noinst_HEADERS = importertest.h exportertest.h + +METASOURCES = AUTO + +LDADD = ../importers/libkrecipesimporters.la ../exporters/libkrecipesexporters.la ../backends/libkrecipesdbs.la ../datablocks/libdatablocks.la $(LIB_TQT) $(LIB_TDEHTML) $(LIB_TDESPELL) + +kretest_SOURCES = kretest.cpp +mmftest_SOURCES = mmftest.cpp +mx2test_SOURCES = mx2test.cpp +mxptest_SOURCES = mxptest.cpp +rezkonvtest_SOURCES = rezkonvtest.cpp +nyctest_SOURCES = nyctest.cpp +recipemltest_SOURCES = recipemltest.cpp diff --git a/src/tests/checks.h b/src/tests/checks.h new file mode 100644 index 0000000..e697814 --- /dev/null +++ b/src/tests/checks.h @@ -0,0 +1,181 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef CHECKS_H +#define CHECKS_H + +#include <cmath> +#include <iostream> + +#include <tqstring.h> +#include <tqpixmap.h> +#include <tqimage.h> + +#include "datablocks/categorytree.h" +#include "datablocks/rating.h" + +using std::cout; +using std::cerr; +using std::endl; + +void check( const RatingList &rating, const RatingList &base ); + +bool check(const TQString &txt, const TQString &a, const TQString &b) +{ + if ( a != b ) { + cout << "ERROR: Tested " << txt.latin1() << ", expected" << endl; + cout << "'" << b.latin1() << "' (" << b.length() << " chars)" << endl; + cout << "but got" << endl; + cout << "'" << a.latin1() << "' (" << a.length() << " chars)" << endl; + exit( 1 ); + } + + return true; +} + +bool check(const TQString &txt, int a, int b) +{ + if ( a != b ) { + cout << "ERROR: Tested " << txt.latin1() << ", expected" << endl; + cout << "'" << b << "'" << endl; + cout << "but got" << endl; + cout << "'" << a << "'" << endl; + exit( 1 ); + } + + return true; +} + +bool check(const TQString &txt, double a, double b) +{ + if ( fabs(a - b) > 1e-10 ) { + cout << "ERROR: Tested " << txt.latin1() << ", expected" << endl; + cout << "'" << b << "'" << endl; + cout << "but got" << endl; + cout << "'" << a << "'" << endl; + exit( 1 ); + } + + return true; +} + +bool check(const TQString &txt, const TQPixmap &a, const TQPixmap &b) +{ + if ( a.size() != b.size() ) { + + cout << "ERROR: Tested " << txt.latin1() << ": photos differ" << endl; + // exit( 1 ); + } + + return true; +} + +void check( const IngredientData &ing, const IngredientData &base_ing, int ing_num ) +{ + check( TQString::number(ing_num)+": Ingredient name", ing.name, base_ing.name ); + check( TQString::number(ing_num)+": Ingredient amount", ing.amount,base_ing.amount ); + check( TQString::number(ing_num)+": Ingredient amount_offset", ing.amount_offset,base_ing.amount_offset ); + check( TQString::number(ing_num)+": Ingredient singular unit", ing.units.name, base_ing.units.name ); + check( TQString::number(ing_num)+": Ingredient plural unit", ing.units.plural, base_ing.units.plural ); + check( TQString::number(ing_num)+": Ingredient group", ing.group, base_ing.group ); + + ElementList::const_iterator prep_it = ing.prepMethodList.begin(); + ElementList::const_iterator base_prep_it = base_ing.prepMethodList.begin(); + for ( ; prep_it != ing.prepMethodList.end(); ++prep_it, ++base_prep_it ) { + check( TQString::number(ing_num)+": Ingredient prep_method", (*prep_it).name, (*base_prep_it).name ); + } +} + +void check( const Recipe &recipe, const Recipe &base ) +{ + check( "Recipe title", recipe.title, base.title ); + check( "Yield base", recipe.yield.amount, base.yield.amount ); + check( "Yield offset", recipe.yield.amount_offset, base.yield.amount_offset ); + check( "Yield type", recipe.yield.type, base.yield.type ); + check( "Instructions", recipe.instructions, base.instructions ); + check( "Photo", recipe.photo, base.photo ); + + check( recipe.ratingList, base.ratingList ); + + int cat_num = 1; + ElementList::const_iterator cat_it = recipe.categoryList.begin(); + ElementList::const_iterator base_cat_it = base.categoryList.begin(); + for ( ; cat_it != recipe.categoryList.end() || base_cat_it != base.categoryList.end(); ++cat_it, ++base_cat_it ) { + check( TQString::number(cat_num)+": Category", (*cat_it).name, (*base_cat_it).name ); + ++cat_num; + } + check( "category count", cat_num-1, base.categoryList.count() ); + + int author_num = 1; + ElementList::const_iterator author_it = recipe.authorList.begin(); + ElementList::const_iterator base_author_it = base.authorList.begin(); + for ( ; author_it != recipe.authorList.end() || base_author_it != base.authorList.end(); ++author_it, ++base_author_it ) { + check( TQString::number(author_num)+": Author", (*author_it).name, (*base_author_it).name ); + ++author_num; + } + check( "author count", author_num-1, base.authorList.count() ); + + int ing_num = 1; + IngredientList::const_iterator ing_it = recipe.ingList.begin(); + IngredientList::const_iterator base_ing_it = base.ingList.begin(); + for ( ; ing_it != recipe.ingList.end() || base_ing_it != base.ingList.end(); ++ing_it, ++base_ing_it ) { + check( *ing_it, *base_ing_it, ing_num ); + + TQValueList<IngredientData>::const_iterator base_sub_it = (*base_ing_it).substitutes.begin(); + for ( TQValueList<IngredientData>::const_iterator sub_it = (*ing_it).substitutes.begin(); sub_it != (*ing_it).substitutes.end(); ++sub_it, ++base_sub_it ) { + check( *sub_it, *base_sub_it, ing_num+1000 ); + } + + ++ing_num; + } + check( "ingredient count", ing_num-1, base.ingList.count() ); +} + +bool check( const CategoryTree *catStructure, const CategoryTree *baseCatStructure ) +{ + CategoryTree * it = catStructure->firstChild(); + CategoryTree * base_it = baseCatStructure->firstChild(); + for ( ; it && base_it; it = it->nextSibling(), base_it = base_it->nextSibling() ) { + check( it, base_it ); + + if ( it->category.name != base_it->category.name ) { + printf("FAILED: Category structure differs\n"); + exit(1); + } + } + + if ( base_it != it ) { //these should both be NULL + printf("FAILED: Category structure differs\n"); + exit(1); + } + + return true; +} + +void check( const RatingList &rating, const RatingList &base ) +{ + RatingList::const_iterator rating_it = rating.begin(); + RatingList::const_iterator base_rating_it = base.begin(); + for ( ; rating_it != rating.end() || base_rating_it != base.end(); ++rating_it, ++base_rating_it ) { + check("checking rater",(*rating_it).rater,(*base_rating_it).rater); + check("checking comment",(*rating_it).comment,(*base_rating_it).comment); + + RatingCriteriaList::const_iterator rc_it = (*rating_it).ratingCriteriaList.begin(); + RatingCriteriaList::const_iterator base_rc_it = (*base_rating_it).ratingCriteriaList.begin(); + for ( ; rc_it != (*rating_it).ratingCriteriaList.end() || base_rc_it != (*base_rating_it).ratingCriteriaList.end(); ++rc_it, ++base_rc_it ) { + check("checking criteria name",(*rc_it).name,(*base_rc_it).name); + check("checking stars",(*rc_it).stars,(*base_rc_it).stars); + } + check( "criteria count", int((*rating_it).ratingCriteriaList.count()), int((*base_rating_it).ratingCriteriaList.count()) ); + } + check( "rating count", int(rating.count()), int(base.count()) ); +} + +#endif diff --git a/src/tests/exportertest.h b/src/tests/exportertest.h new file mode 100644 index 0000000..739a57c --- /dev/null +++ b/src/tests/exportertest.h @@ -0,0 +1,42 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef EXPORTERTEST_H +#define EXPORTERTEST_H + +#include <cmath> +#include <iostream> + +#include <tqstring.h> +#include <tqfile.h> +#include <tqtextstream.h> + +#include "checks.h" + +using std::cout; +using std::cerr; +using std::endl; + +void check( BaseExporter &exporter, const RecipeList &recipeList ) +{ + TQFile file("test.txt"); + if ( file.open( IO_WriteOnly ) ) { + TQTextStream stream(&file); + exporter.writeStream(stream,recipeList); + } + else { + printf("Unable to open file for writing\n"); + exit(1); + } + + file.close(); +} + +#endif diff --git a/src/tests/importertest.h b/src/tests/importertest.h new file mode 100644 index 0000000..f5aff84 --- /dev/null +++ b/src/tests/importertest.h @@ -0,0 +1,45 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef IMPORTERTEST_H +#define IMPORTERTEST_H + +#include <cmath> +#include <iostream> + +#include <tqstring.h> + +#include "checks.h" +#include "datablocks/categorytree.h" + +using std::cout; +using std::endl; + +void check( const BaseImporter &importer, const Recipe &recipe ) +{ + int recipe_num = 1; + RecipeList recipeList = importer.recipeList(); + for ( RecipeList::const_iterator it = recipeList.begin(); it != recipeList.end(); ++it ) { + printf("Recipe %d... ",recipe_num); + check( *it, recipe ); + printf("successful\n"); + ++recipe_num; + } + + check( "recipe count", recipe_num-1, 2 ); +} + +void check( const BaseImporter &importer, const CategoryTree *baseCatStructure ) +{ + printf("Checking category structure.\n"); + check( importer.categoryStructure(), baseCatStructure ); +} + +#endif diff --git a/src/tests/kretest.cpp b/src/tests/kretest.cpp new file mode 100644 index 0000000..10c88c2 --- /dev/null +++ b/src/tests/kretest.cpp @@ -0,0 +1,191 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include <tdeapplication.h> + +#include <tqstring.h> +#include <tqfile.h> + +#include <iostream> + +#include "kreimporter.h" +#include "kreexporter.h" +#include "importertest.h" +#include "exportertest.h" + +int +main(int argc, char *argv[]) +{ + TDEApplication a(argc, argv, "kretest"); + + printf("Creating KreImporter.\n"); + KreImporter importer; + + printf("Parsing kretest.txt.\n"); + TQStringList files; files << "kretest.txt"; + importer.parseFiles(files); + + Recipe recipe; + recipe.title = "Cookies_Test"; + recipe.yield.amount = 2; + recipe.yield.amount_offset = 1; + recipe.yield.type = "dozen"; + recipe.categoryList.append( Element("Snacks",1) ); + recipe.categoryList.append( Element("Cookies & Squares",2) ); + recipe.instructions = + "Drop by spoonful on greased cookie sheet. Bake in moderate oven."; + recipe.prepTime = TQTime(0,30); + if ( !recipe.photo.load( "test_photo.jpg", "JPEG" ) ) { + printf("Unable to load test_photo.jpg\n"); + exit(1); + } + + recipe.authorList.append( Element("Colleen Beamer") ); + recipe.authorList.append( Element("Mona Beamer") ); + + Ingredient ing; + ing.name = "granulated sugar"; + ing.amount = 0.75; + ing.amount_offset = 0.25; + ing.units.name = "c."; + ing.groupID = 0; ing.group = "Dry Ingredients"; + recipe.ingList.append( ing ); + + Ingredient ing2; + ing2.name = "brown sugar"; + ing2.amount = 1; + ing2.amount_offset = 0; + ing2.units.name = "c."; + ing2.groupID = 0; ing2.group = "Dry Ingredients"; + recipe.ingList.append( ing2 ); + + Ingredient ing3; + ing3.name = "all-purpose flour"; + ing3.amount = 2; + ing3.units.plural = "c."; + ing3.groupID = 0; ing3.group = "Dry Ingredients"; + recipe.ingList.append( ing3 ); + + Ingredient ing4; + ing4.name = "baking soda"; + ing4.amount = 1; + ing4.amount_offset = 0; + ing4.units.name = "tsp."; + ing4.groupID = 0; ing4.group = "Dry Ingredients"; + recipe.ingList.append( ing4 ); + + Ingredient ing8; + ing8.name = "shortening"; + ing8.amount = 1; + ing8.amount_offset = 0; + ing8.units.name = "c."; + ing8.prepMethodList.append( Element("softened") ); + ing8.prepMethodList.append( Element("at room temperature") ); + ing8.groupID = 1; ing8.group = "Fat & Liquids"; + recipe.ingList.append( ing8 ); + + Ingredient ing6; + ing6.name = "peanut butter"; + ing6.amount = 1; + ing6.amount_offset = 0; + ing6.units.name = "c."; + ing6.groupID = 1; ing6.group = "Fat & Liquids"; + recipe.ingList.append( ing6 ); + + Ingredient ing5; + ing5.name = "eggs"; + ing5.amount = 2; + ing5.amount_offset = 0; + ing5.units.plural = ""; + ing5.groupID = 1; ing5.group = "Fat & Liquids"; + recipe.ingList.append( ing5 ); + + Ingredient ing7; + ing7.name = "vanilla extract"; + ing7.amount = 1; + ing7.amount_offset = 0; + ing7.units.name = "tsp."; + ing7.groupID = 1; ing7.group = "Fat & Liquids"; + recipe.ingList.append( ing7 ); + + Ingredient ing9; + ing9.name = "a"; + ing9.amount = 1; + ing9.amount_offset = 0; + ing9.units.name = "cup"; + IngredientData ing9_1; + ing9_1.name = "b"; + ing9_1.amount = 2; + ing9_1.amount_offset = 0; + ing9_1.units.plural = "cups"; + IngredientData ing9_2; + ing9_2.name = "c"; + ing9_2.amount = 3; + ing9_2.amount_offset = 0; + ing9_2.units.plural = "cups"; + ing9.substitutes.append(ing9_1); + ing9.substitutes.append(ing9_2); + recipe.ingList.append( ing9 ); + + CategoryTree *catTree = new CategoryTree; + (void)catTree->add( Element("Cookies & Squares",2) ); + (void)catTree->add( Element("Snacks",1) ); + + RatingCriteria rc; + Rating rating1; + rating1.rater = "George McFry"; + rating1.comment = "Good enough"; + + rc.name = "Taste"; + rc.stars = 5.0; + rating1.append(rc); + + Rating rating2; + rating2.rater = "Me"; + rating2.comment = "Yuck, don't eat!"; + + rc.name = "Overall"; + rc.stars = 2.0; + rating2.append(rc); + + rc.name = "Taste"; + rc.stars = 1.5; + rating2.append(rc); + + recipe.ratingList.append(rating1); + recipe.ratingList.append(rating2); + + + check( importer, recipe ); + check( importer, catTree ); + + + RecipeList recipeList; + recipeList.append(recipe); + recipeList.append(recipe); + + printf("Creating KreExporter.\n"); + KreExporter exporter(catTree,"not needed",".kreml"); + check( exporter, recipeList ); + printf("Successfully exported recipes to test.txt.\n"); + + printf("Creating KreImporter to test exported recipes.\n"); + KreImporter importer2; + + printf("Parsing test.txt.\n"); + TQStringList files2; files2 << "test.txt"; + importer2.parseFiles(files2); + TQFile::remove("test.txt"); + check( importer2, recipe ); + check( importer2, catTree ); + printf("Recipe export successful.\n"); + + printf("*** Krecipes importer and exporter passed the tests :-) ***\n"); +} diff --git a/src/tests/kretest.txt b/src/tests/kretest.txt new file mode 100644 index 0000000..35b4ae6 --- /dev/null +++ b/src/tests/kretest.txt @@ -0,0 +1,409 @@ +<?xml version="1.0" encoding="UTF-8" ?> +<krecipes version="SVN_PRE-0.9" lang="en_US" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="krecipes.xsd"> +<krecipes-category-structure> +<category name="Cookies & Squares"> +</category> +<category name="Snacks"> +</category> +</krecipes-category-structure> +<krecipes-recipe> +<krecipes-description> +<title>Cookies_Test</title> +<author>Colleen Beamer</author> +<author>Mona Beamer</author> +<pictures> +<pic format="JPEG" id="1"><![CDATA[/9j/4AAQSkZJRgABAQIAAAAAAAD/2wBDAAgGBgcGBQgHBwcJCQgKDBQNDAsLDBkSEw8UHRofHh0a +HBwgJC4nICIsIxwcKDcpLDAxNDQ0Hyc5PTgyPC4zNDL/2wBDAQkJCQwLDBgNDRgyIRwhMjIyMjIy +MjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjL/wAARCACmAIMDASIA +AhEBAxEB/8QAHwAAAQUBAQEBAQEAAAAAAAAAAAECAwQFBgcICQoL/8QAtRAAAgEDAwIEAwUFBAQA +AAF9AQIDAAQRBRIhMUEGE1FhByJxFDKBkaEII0KxwRVS0fAkM2JyggkKFhcYGRolJicoKSo0NTY3 +ODk6Q0RFRkdISUpTVFVWV1hZWmNkZWZnaGlqc3R1dnd4eXqDhIWGh4iJipKTlJWWl5iZmqKjpKWm +p6ipqrKztLW2t7i5usLDxMXGx8jJytLT1NXW19jZ2uHi4+Tl5ufo6erx8vP09fb3+Pn6/8QAHwEA +AwEBAQEBAQEBAQAAAAAAAAECAwQFBgcICQoL/8QAtREAAgECBAQDBAcFBAQAAQJ3AAECAxEEBSEx +BhJBUQdhcRMiMoEIFEKRobHBCSMzUvAVYnLRChYkNOEl8RcYGRomJygpKjU2Nzg5OkNERUZHSElK +U1RVVldYWVpjZGVmZ2hpanN0dXZ3eHl6goOEhYaHiImKkpOUlZaXmJmaoqOkpaanqKmqsrO0tba3 +uLm6wsPExcbHyMnK0tPU1dbX2Nna4uPk5ebn6Onq8vP09fb3+Pn6/9oADAMBAAIRAxEAPwD2iyso +Z7cO6kscg4NWRpluR0b/AL6o0z/jzH1Nc5PcTCeQCVwNx6MfWtZSlzPUzjGPKdH/AGZb+jfnR/Zl +v6N+dcx9om/57Sf99Gj7RN/z2k/76NRzz7lcsTp/7Lt/Rvzo/sy39G/OuZ+0zAZ81j7FzWXceKdO +tJXgn1mGOaM/OHnA2/X0o559w5Yndf2Zb+jfnR/Zlv6N/wB9VyVrqiX1ulzaXvn27EfvUkypGecH +oam8+cZzNJ9Cx/n/AJ60c8+4csTp/wCzLf0b/vqj+zLf0b865j7RNkjzpAR1G40faJv+e0n/AH0a +OefcOWJ0/wDZlv6N+dH9mW/o351zH2ib/ntJ/wB9Gj7RN/z2k/76NHPPuHLE6f8Asy39G/Oj+zLf +0b865j7RN/z2k/76NH2ib/ntJ/30aOefcOWJ050y39G/76pP7Mt/Rv8AvquZ+0Tf89pP++jXSaOz +Pp6sxLHJ5Jo55dw5FbQx5PLSRlA4BopJ/wDXv/vGiu3UxNzTP+PMfU1zE/8Ax8S/75/nXT6Z/wAe +g+prmJ/+PiX/AHz/ADrin8bNov3SOjHU5wByaKOvHrxUlGR4pu2svDl7KqCRigjxyMbyF6/jmvL4 +YtqooJzyd3fNd748lmGl28aufLmnCygjrgFh/IVyltBuI46UAdn4EsENhdyyxq24heR781LrCyaX +vlsJPIfnCfwNnGcj8BzW14TtfK8PhsH5yW/WsTxXJstZSDxtOT/h+VJtIFG+xe0bV7bWtNiureVJ +DjbIsbZ2uOGHODwQRk9R7Vod/ryPpXkXwc1OVr/WrDZF5Jf7QJMHO4tt9emK9d6KCBgMTyOnH/66 +B8rfQKKP5etGecd6buK3YKKO4HrQDkZoE2ugV0+i/wDINX/eNcxXT6L/AMg1f940gMif/Xyf7xoo +n/18n+8aK70jlubmmf8AHmPqa5ef/j4l/wB8/wA66jTP+PMfU1y8/wDx8S/75/nXHP42dMfhI6Oc +cfn6UUf0qSjifGUhk1S0iWXciRsxTPfPGfwNV9PtSyZHek1gR3Hie5aJkcDaj7TyGUbT+oro9N0/ +EMZ28HrQB0Mug3OoeFreCzv5rKdIiVaNiMttxzjtmvGvFia9ocbjUdVNyzAkFJWI445yPevoiJlg +0zOMBEx+lfP/AMUJvPQkD+F+B9V5/Ssaq0PTy+rLmcXroyr8NBr14ZLrSriztdPS5VbuJ87nHBO0 +AYyRxmvSpfFehw3Rt3v41mztKhGOD7HGPxzXmPgn7XYfDTVrlDJD9pul8t8Ab1OFODj6j/CszTtV +8PWYmOrWct9cOwKnziqoPQAfrnPQVEpuLsjSjhYVqbq1Nr2PdEuYJLf7QkyNDgsZVYFSB15rIHjH +QS4i/tJd3UAowz29PevKdEvZ9Y1250jTHK297GxETO21duWCrgjGeAfXJrTtdN0fQ3Fn4k0aWaUy +PtuvMcBlGOFAI3AU3VldMcMDRtJLVnYeK/FjaStq2mSwXDOGfaG3K6DsG7ZIIrT0PxPY61bQkzwR +XjIPMtvMyyNjlQTjdj2rifGcWkw+DtNfRURLHZKY9rZOOSRk5PDbsjPXNbfgrwroi6LpGtiyX+0D +AsjXDStnJXDHBOOc+lOnJuTuZ4iFKNKPKtWdvXT6L/yDV/3jXMkd/mPoT0/Sul0b/kGr/vGt2eVf +cyZ/9fJ/vGiif/Xyf7xoruRym5pn/HmPqa5ef/j4l/3z/Ouo0z/jzH1NcvP/AMfEv++f51xz+NnT +H4SOkYlVY7ScjGAM5payvE2oS6V4Z1G9gVDLDAzKHBI/Qg/rUlHDR67pKatcSXF2sBad3aOT7yEs +cggd88V6I+q2VnoYvVWRrdUBLqhOa+TTM7Tea7szM25mY5JOeTnvXumj+N767FpZ2tvbSaLJa7Zy +4+ZW6Hv9KTaSuwNe4+M/hqa0aAXUyFgQQYm4rmNb0y98V+HJtX0tPPtlUgclWPToD1qtZfCH+0Yp +ruS+SNZizxoEzgHp/Stu2uPEHh3RrXRLVraAQO3nmRfvx8YI/X9K4oY6hVn7KL1N6M505c0RPhk0 +Ws+BZtFu7ZTFaTPCQx+9uJfJHUEFuMelR6jHr3gOwk/s9oZdOaVpAVC7lJx1DY5PbbnoarfDG+J8 +Y+I7WKX/AELcZtmBgtuwTntx+FUte1K58T6swMmbKEkQRr8oH+0R612TppvQulip04uKV0ULr4h+ +I9TiSLm3i2kM0JCyOQScgjpxgY7496zZtd1ySJ42uNTuAw5EtwWX8ia7DSvCpnI+TJyOw6V2Vl8P +YpoxuO0+1S6a6FU8ZODvG1zwy0N1FbyQTbvJmcsY9/AJ4Jx68fpXo3w/8Tapcz22jw2ay2MEYDSu +yI8KAYBABJYE4B9OK1Nf+Hpsoi8QDKB6V5rey3vhy6TUbCUw3Fs2QcfeB42Edwc9KajYmWLnUjyz +R9CEdCRyR6jgfQf55rptF/5Bq/7xri9G1BNV0a0vY3RxNErsUHG8j5gPYHI/Cu00X/kGr9TVs5DI +n/18n+8aKJ/9fJ/vGiu5HMbmmf8AHmPqa5ef/j4l/wB8/wA66jTP+PMfU1y8/wDx8S/75/nXHP42 +dEfhI6p6rp0OraVdWNwrNDNGVYKcEj2q5QM5BGOPWpKPka9tJ7C8ntLmMRz27mOVcg7WU4I9Dz6V +HFdyw/6uR15zgHFegfGLSVsfFYu0dNt7EH2KmMEYBye5J5rzemgPY/AXxDw8dlfylc4XJ6Gu+8X+ +H4fFWhEw488AtE4PKnHb618xpI0bhkJDDkEdq9X+H/xIa3kGnam6lHwFkbtXzWYZXOlU+tYbdbm0 +ZrZlH4eRXGmSeLF3PBcW+my4ZTgxuucEe/GeK6bw1apcXsEMnO/GPQ+4/wDr81tXehWlr48sNdjl +RLS8VoZ1Y/I7MhVcjvkkD8ar2el3WjavFHOwLK29JFGA4z/9fpXt4TEKvTUlv1M5KzPRk0+KwcIq +j5QM4rUt7hEUZNZc92s4WUH7wANQGcgHJwRXSSzR1C5WWN0Azxz7V4L8SUjt7OQ7gN8gC4HU16xq +epRWlrJNPKI0UfMSccV4B4k1a78aeI4rHTojIGk2Qxr/ABH1/IE0Aex/DL/knek8bRiQgf8AbRua +9U0X/kGr9TXB+GtGGgeHLLTVZmEMeGLHPzElmA9gSa7zRf8AkGr9TQBkT/6+T/eNFE/+vk/3jRXc +jmNzTP8AjzH1NcvP/wAfEv8Avn+ddRpn/HmPqa5ef/j4l/3z/OuOfxs3j8JHR05xnHOPWjoM8cet +Z+s6xZ6Fpsl9eSBI06A8lj6D1PtUlHlXxtvoDPp1isUBmVS5kUgugP8ABjsvOfwFeZJo8rWS3O4B +Sam1jVL3xJrkl/dsJLmdxwgwFHQKB6DgV7b4A+GUN3pVvc6wHZWOVg6A/WgDwy/0prO1jnMgYPg8 +e9O0SwGqaxZ2HnxxCaUAvK21F/HtX11ceANAubL7LJpcAj242qpJH45rxDx58Lx4fkXU9EDvFE+5 +4H5Ckc5yMce1AHpNp4eOnpHBa3zmyjeN4oHjDlNuDjc3PJH4ZrVvLKC9t/ImB2ZypDfMvuD2/wD1 +1n+F9bTxD4cstSUAPLGBKApAVxwwAJPGQcVqyyLDE8jZwiljj0AzWdOnCm24Kw27nJX+pX+gjUHL +rcwwlViChvNJIBC9OTzwQT271zep/EHxDbySQReFrxJ0O0Exs6g98MAc1s6A03iLUxdTFns4JmuU +Zx94k/IMjABUbT0Nd1WgjxC78NfEHxjcL9viW2tpE3J5sqhFGMgYUk5/Cu78HfDrTfDCw3cyi51R +UKmUnKoT12encZ4yCa7SigAwMnGCcD5sYz7V0+i/8g1fqa5iun0X/kGr9TQBkT/6+T/eNFE/+vk/ +3jRXcjmNzTP+PMfU1y8//HxL/vn+ddRpn/HmPqa5e4/4+ZMf3z/OuOfxs3j8JGOXAHTnP5ZrxL4w +eJftV/HoltKTDbDdOoJAMp7FehwNpz/tGvWPEurx6B4evNTcY8lPlGMgsTheOO5GfbNfLN7dzX15 +Ncztuklcux9z9ako6f4c6SNW8UQGXlITvbP+fevrmLy7LSo1XGMAYFfMfwr2wzyzHGS2K9wl1hpY +EXfxQB2MV/AYgTgH0rk/EU8dwzggMCMHjtVM6kyrw1Zl5eljkkE4PNAHAWPiFPAHiy4sp0P9jahI +JWdFyYmPG73A7+3SvQdcubHU9HltYriGWKaPzGKncPLHO788fr6V5P8AEZVls45QOUbt6VJZzHwt +8N0uZt0lzqEgXY7Y/dDnA9KAPUvBlh9j0FJmiaOa5kaZvnDArn5CCDxlAvFdDVHRpobrQbKe2iVI +Ht4zGgfOwbRxnuR0/Cr1ABRRRQAV0+i/8g1fqa5iun0X/kGr9TQBkT/6+T/eNFE/+vk/3jRXcjmN +zTP+PMfU1y8//HxLn+8T+tdRpn/HmPqa5LUruCyS6uriRY4YQ7SOx2gAe9cc/jZvH4Tyv41ahZtp +VlpvnobtbjzWjxyF2sOT9SP514k3KjGce9dd4ih1rxHDfeLprWQWDzbA552rnC9hlRwufUiuRkzn +kYNSUdt4Dv8AypXQtjnNeqW+oZjBLDFeBaNfGw1COUfdJwa9Q0/VFkiB3DBoA7Jrvg81SuLvPesr +7eCvUVTur8KnXr3oAq6rZNrl3b6ep4mmRWPouRn9K574kaql5rUNjbsDb2ibFC9AaivfFU9lczra +riRlZBKf4cjGa5uxs7rWNWgtoRJLPcOFBA3H3OPzoA+h/hlHLF8PdLWZGVwsrbHB3YaRiDj0IOfx +zXXVU0u0+waTZ2RcP9mhWJm6btowD7dOmTVugAooooAK6fRf+Qav1NcxXT6L/wAg1fqaAMif/Xyf +7xoon/18n+8aK7kcxt6Z/wAeY+p4rzTxvpGo6/HHpNo5hs7iZjdziTBjQfwhc87vy4r0zTP+PMfU +1y8//HxL/vn+dcc/jZvH4TIn0Cxm8OHRPL/0EQCBRIN+wAYU4PXBwR7gV83+K/C2oeFdTa0vEJjb +JhnH3ZF9vQ+oPNfUfAPt39qy9e0vSdU09k1m2ilt0+YeZnIPqCCKhtRV2Uj5OGQcit/TdaaFFRmI +ArZ1DwDdLPO9hvkt97eSJR8xQHjJ9cda5jUtJutLkVLiMqXGV96zhXpVHyqWo7HULr8ZXmSqV3rq +Mv38iuetrdp3CAhcn7zHH4D1rqLPwcktuxnuCZOPuD7v+NFWrCkryYJXE1zXdKufDVlZ2KD7SBi5 +d48MT9cV6n8NfAjeG4Tql7ldTuE2hFbIhjODt9ycD6Y964Lwz4Pht/FunNf3Aax84MMYBZgcopyD +wWAH4176ORn26g8H8KqnVhNaMT0F4z069j/D6GkoxRVgFFFFABXT6L/yDV+prmK6fRf+Qav1NAGR +P/r5P940UT/6+T/eNFdyOY3NM/48x9TXLz/8fEv++f510+mf8eY+prJl0W7eV2Hl4LEj5q45/Gze +PwmV3rm/Gekalq2kImmyBpIn8w27EATEdPmPAI7ZIHNdt/YV5/0z/wC+qX+xLz/pn9N3Ws5RTVmV +c8igsvF9uqINCkkGMYe5g5H4vxTL/wAH6v4tmt4tRsk0y2hYFmLqzyKc7tuwnGMDrjrXr/8AYd5n +P7vPTlu1B0O87bM+u7oPTpXLTwdKnPnitR82hxtv4J8PW2j/ANlJpsLW2S2XG5gxyNwY87gDgHqO +1Yq/DaK2uXey1m6ghPCRMgkCD05r03+wrvsI8f71J/Yd5jGI/wDvqumUIz+JXEmzi7LwVpVpc290 +4muLiHBDSysV3gYLbc4BzzXRkkkkk89a0f7CvB/zz/76o/sK8/6Z/wDfVOMVH4UBm0Vpf2Fef9M/ +++qP7CvP+mf/AH1VXAzaK0v7CvP+mf8A31R/YV5/0z/76ouBm10+i/8AINX6msr+w7z/AKZ/99Vt +abbva2YikxuBJ4NAGHP/AK+T/eNFE/8Ar35H3jRXcjmLlveSW8YiCKcE8kmpf7XcceUv50UVEoRv +sXFsP7Yf/nkv50n9sP8A88l/OiilyR7Duw/tiT/nkv50f2xJ/wA8l/Oiijkj2C7D+2JP+eS/nR/b +En/PJfzooo5I9guw/tiT/nkv50f2xJ/zyX86KKOSPYLsP7Yk/wCeS/nR/bEn/PJfzooo5I9guw/t +iT/nkv50f2xJ/wA8l/Oiijkj2C7F/th/+eS/nR/a7n/lkv50UUKEewXZmtJudjtHJooorexkf//Z]]></pic> +</pictures> +<category> +<cat>Snacks</cat> +<cat>Cookies & Squares</cat> +</category> +<yield><amount><min>2</min><max>3</max></amount><type>dozen</type></yield> +<preparation-time>00:30</preparation-time> +</krecipes-description> +<krecipes-ingredients> +<ingredient-group name="Dry Ingredients"> +<ingredient> +<name>granulated sugar</name> +<amount><min>0.75</min><max>1</max></amount> +<unit>c.</unit> +</ingredient> +<ingredient> +<name>brown sugar</name> +<amount>1</amount> +<unit>c.</unit> +</ingredient> +<ingredient> +<name>all-purpose flour</name> +<amount>2</amount> +<unit>c.</unit> +</ingredient> +<ingredient> +<name>baking soda</name> +<amount>1</amount> +<unit>tsp.</unit> +</ingredient> +</ingredient-group> +<ingredient-group name="Fat & Liquids"> +<ingredient> +<name>shortening</name> +<amount>1</amount> +<unit>c.</unit> +<prep>softened,at room temperature</prep> +</ingredient> +<ingredient> +<name>peanut butter</name> +<amount>1</amount> +<unit>c.</unit> +</ingredient> +<ingredient> +<name>eggs</name> +<amount>2</amount> +<unit></unit> +</ingredient> +<ingredient> +<name>vanilla extract</name> +<amount>1</amount> +<unit>tsp.</unit> +</ingredient> +</ingredient-group> +<ingredient> + <name>a</name> + <amount>1</amount> + <unit>cup</unit> + <substitutes> + <ingredient> + <name>b</name> + <amount>2</amount> + <unit>cups</unit> + </ingredient> + <ingredient> + <name>c</name> + <amount>3</amount> + <unit>cups</unit> + </ingredient> + </substitutes> +</ingredient> +</krecipes-ingredients> +<krecipes-instructions> +Drop by spoonful on greased cookie sheet. Bake in moderate oven.</krecipes-instructions> +<krecipes-ratings> + <rating> + <rater>George McFry</rater> + <comment>Good enough</comment> + <criterion> + <criteria> + <name>Taste</name> + <stars>5</stars> + </criteria> + </criterion> + </rating> + <rating> + <rater>Me</rater> + <comment>Yuck, don't eat!</comment> + <criterion> + <criteria> + <name>Overall</name> + <stars>2</stars> + </criteria> + <criteria> + <name>Taste</name> + <stars>1.5</stars> + </criteria> + </criterion> + </rating> +</krecipes-ratings> +</krecipes-recipe> +<krecipes-recipe> +<krecipes-description> +<title>Cookies_Test</title> +<author>Colleen Beamer</author> +<author>Mona Beamer</author> +<pictures> +<pic format="JPEG" id="1"><![CDATA[/9j/4AAQSkZJRgABAQIAAAAAAAD/2wBDAAgGBgcGBQgHBwcJCQgKDBQNDAsLDBkSEw8UHRofHh0a +HBwgJC4nICIsIxwcKDcpLDAxNDQ0Hyc5PTgyPC4zNDL/2wBDAQkJCQwLDBgNDRgyIRwhMjIyMjIy +MjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjL/wAARCACmAIMDASIA +AhEBAxEB/8QAHwAAAQUBAQEBAQEAAAAAAAAAAAECAwQFBgcICQoL/8QAtRAAAgEDAwIEAwUFBAQA +AAF9AQIDAAQRBRIhMUEGE1FhByJxFDKBkaEII0KxwRVS0fAkM2JyggkKFhcYGRolJicoKSo0NTY3 +ODk6Q0RFRkdISUpTVFVWV1hZWmNkZWZnaGlqc3R1dnd4eXqDhIWGh4iJipKTlJWWl5iZmqKjpKWm +p6ipqrKztLW2t7i5usLDxMXGx8jJytLT1NXW19jZ2uHi4+Tl5ufo6erx8vP09fb3+Pn6/8QAHwEA +AwEBAQEBAQEBAQAAAAAAAAECAwQFBgcICQoL/8QAtREAAgECBAQDBAcFBAQAAQJ3AAECAxEEBSEx +BhJBUQdhcRMiMoEIFEKRobHBCSMzUvAVYnLRChYkNOEl8RcYGRomJygpKjU2Nzg5OkNERUZHSElK +U1RVVldYWVpjZGVmZ2hpanN0dXZ3eHl6goOEhYaHiImKkpOUlZaXmJmaoqOkpaanqKmqsrO0tba3 +uLm6wsPExcbHyMnK0tPU1dbX2Nna4uPk5ebn6Onq8vP09fb3+Pn6/9oADAMBAAIRAxEAPwD2iyso +Z7cO6kscg4NWRpluR0b/AL6o0z/jzH1Nc5PcTCeQCVwNx6MfWtZSlzPUzjGPKdH/AGZb+jfnR/Zl +v6N+dcx9om/57Sf99Gj7RN/z2k/76NRzz7lcsTp/7Lt/Rvzo/sy39G/OuZ+0zAZ81j7FzWXceKdO +tJXgn1mGOaM/OHnA2/X0o559w5Yndf2Zb+jfnR/Zlv6N/wB9VyVrqiX1ulzaXvn27EfvUkypGecH +oam8+cZzNJ9Cx/n/AJ60c8+4csTp/wCzLf0b/vqj+zLf0b865j7RNkjzpAR1G40faJv+e0n/AH0a +OefcOWJ0/wDZlv6N+dH9mW/o351zH2ib/ntJ/wB9Gj7RN/z2k/76NHPPuHLE6f8Asy39G/Oj+zLf +0b865j7RN/z2k/76NH2ib/ntJ/30aOefcOWJ050y39G/76pP7Mt/Rv8AvquZ+0Tf89pP++jXSaOz +Pp6sxLHJ5Jo55dw5FbQx5PLSRlA4BopJ/wDXv/vGiu3UxNzTP+PMfU1zE/8Ax8S/75/nXT6Z/wAe +g+prmJ/+PiX/AHz/ADrin8bNov3SOjHU5wByaKOvHrxUlGR4pu2svDl7KqCRigjxyMbyF6/jmvL4 +YtqooJzyd3fNd748lmGl28aufLmnCygjrgFh/IVyltBuI46UAdn4EsENhdyyxq24heR781LrCyaX +vlsJPIfnCfwNnGcj8BzW14TtfK8PhsH5yW/WsTxXJstZSDxtOT/h+VJtIFG+xe0bV7bWtNiureVJ +DjbIsbZ2uOGHODwQRk9R7Vod/ryPpXkXwc1OVr/WrDZF5Jf7QJMHO4tt9emK9d6KCBgMTyOnH/66 +B8rfQKKP5etGecd6buK3YKKO4HrQDkZoE2ugV0+i/wDINX/eNcxXT6L/AMg1f940gMif/Xyf7xoo +n/18n+8aK70jlubmmf8AHmPqa5ef/j4l/wB8/wA66jTP+PMfU1y8/wDx8S/75/nXHP42dMfhI6Oc +cfn6UUf0qSjifGUhk1S0iWXciRsxTPfPGfwNV9PtSyZHek1gR3Hie5aJkcDaj7TyGUbT+oro9N0/ +EMZ28HrQB0Mug3OoeFreCzv5rKdIiVaNiMttxzjtmvGvFia9ocbjUdVNyzAkFJWI445yPevoiJlg +0zOMBEx+lfP/AMUJvPQkD+F+B9V5/Ssaq0PTy+rLmcXroyr8NBr14ZLrSriztdPS5VbuJ87nHBO0 +AYyRxmvSpfFehw3Rt3v41mztKhGOD7HGPxzXmPgn7XYfDTVrlDJD9pul8t8Ab1OFODj6j/CszTtV +8PWYmOrWct9cOwKnziqoPQAfrnPQVEpuLsjSjhYVqbq1Nr2PdEuYJLf7QkyNDgsZVYFSB15rIHjH +QS4i/tJd3UAowz29PevKdEvZ9Y1250jTHK297GxETO21duWCrgjGeAfXJrTtdN0fQ3Fn4k0aWaUy +PtuvMcBlGOFAI3AU3VldMcMDRtJLVnYeK/FjaStq2mSwXDOGfaG3K6DsG7ZIIrT0PxPY61bQkzwR +XjIPMtvMyyNjlQTjdj2rifGcWkw+DtNfRURLHZKY9rZOOSRk5PDbsjPXNbfgrwroi6LpGtiyX+0D +AsjXDStnJXDHBOOc+lOnJuTuZ4iFKNKPKtWdvXT6L/yDV/3jXMkd/mPoT0/Sul0b/kGr/vGt2eVf +cyZ/9fJ/vGiif/Xyf7xoruRym5pn/HmPqa5ef/j4l/3z/Ouo0z/jzH1NcvP/AMfEv++f51xz+NnT +H4SOkYlVY7ScjGAM5payvE2oS6V4Z1G9gVDLDAzKHBI/Qg/rUlHDR67pKatcSXF2sBad3aOT7yEs +cggd88V6I+q2VnoYvVWRrdUBLqhOa+TTM7Tea7szM25mY5JOeTnvXumj+N767FpZ2tvbSaLJa7Zy +4+ZW6Hv9KTaSuwNe4+M/hqa0aAXUyFgQQYm4rmNb0y98V+HJtX0tPPtlUgclWPToD1qtZfCH+0Yp +ruS+SNZizxoEzgHp/Stu2uPEHh3RrXRLVraAQO3nmRfvx8YI/X9K4oY6hVn7KL1N6M505c0RPhk0 +Ws+BZtFu7ZTFaTPCQx+9uJfJHUEFuMelR6jHr3gOwk/s9oZdOaVpAVC7lJx1DY5PbbnoarfDG+J8 +Y+I7WKX/AELcZtmBgtuwTntx+FUte1K58T6swMmbKEkQRr8oH+0R612TppvQulip04uKV0ULr4h+ +I9TiSLm3i2kM0JCyOQScgjpxgY7496zZtd1ySJ42uNTuAw5EtwWX8ia7DSvCpnI+TJyOw6V2Vl8P +YpoxuO0+1S6a6FU8ZODvG1zwy0N1FbyQTbvJmcsY9/AJ4Jx68fpXo3w/8Tapcz22jw2ay2MEYDSu +yI8KAYBABJYE4B9OK1Nf+Hpsoi8QDKB6V5rey3vhy6TUbCUw3Fs2QcfeB42Edwc9KajYmWLnUjyz +R9CEdCRyR6jgfQf55rptF/5Bq/7xri9G1BNV0a0vY3RxNErsUHG8j5gPYHI/Cu00X/kGr9TVs5DI +n/18n+8aKJ/9fJ/vGiu5HMbmmf8AHmPqa5ef/j4l/wB8/wA66jTP+PMfU1y8/wDx8S/75/nXHP42 +dEfhI6p6rp0OraVdWNwrNDNGVYKcEj2q5QM5BGOPWpKPka9tJ7C8ntLmMRz27mOVcg7WU4I9Dz6V +HFdyw/6uR15zgHFegfGLSVsfFYu0dNt7EH2KmMEYBye5J5rzemgPY/AXxDw8dlfylc4XJ6Gu+8X+ +H4fFWhEw488AtE4PKnHb618xpI0bhkJDDkEdq9X+H/xIa3kGnam6lHwFkbtXzWYZXOlU+tYbdbm0 +ZrZlH4eRXGmSeLF3PBcW+my4ZTgxuucEe/GeK6bw1apcXsEMnO/GPQ+4/wDr81tXehWlr48sNdjl +RLS8VoZ1Y/I7MhVcjvkkD8ar2el3WjavFHOwLK29JFGA4z/9fpXt4TEKvTUlv1M5KzPRk0+KwcIq +j5QM4rUt7hEUZNZc92s4WUH7wANQGcgHJwRXSSzR1C5WWN0Azxz7V4L8SUjt7OQ7gN8gC4HU16xq +epRWlrJNPKI0UfMSccV4B4k1a78aeI4rHTojIGk2Qxr/ABH1/IE0Aex/DL/knek8bRiQgf8AbRua +9U0X/kGr9TXB+GtGGgeHLLTVZmEMeGLHPzElmA9gSa7zRf8AkGr9TQBkT/6+T/eNFE/+vk/3jRXc +jmNzTP8AjzH1NcvP/wAfEv8Avn+ddRpn/HmPqa5ef/j4l/3z/OuOfxs3j8JHR05xnHOPWjoM8cet +Z+s6xZ6Fpsl9eSBI06A8lj6D1PtUlHlXxtvoDPp1isUBmVS5kUgugP8ABjsvOfwFeZJo8rWS3O4B +Sam1jVL3xJrkl/dsJLmdxwgwFHQKB6DgV7b4A+GUN3pVvc6wHZWOVg6A/WgDwy/0prO1jnMgYPg8 +e9O0SwGqaxZ2HnxxCaUAvK21F/HtX11ceANAubL7LJpcAj242qpJH45rxDx58Lx4fkXU9EDvFE+5 +4H5Ckc5yMce1AHpNp4eOnpHBa3zmyjeN4oHjDlNuDjc3PJH4ZrVvLKC9t/ImB2ZypDfMvuD2/wD1 +1n+F9bTxD4cstSUAPLGBKApAVxwwAJPGQcVqyyLDE8jZwiljj0AzWdOnCm24Kw27nJX+pX+gjUHL +rcwwlViChvNJIBC9OTzwQT271zep/EHxDbySQReFrxJ0O0Exs6g98MAc1s6A03iLUxdTFns4JmuU +Zx94k/IMjABUbT0Nd1WgjxC78NfEHxjcL9viW2tpE3J5sqhFGMgYUk5/Cu78HfDrTfDCw3cyi51R +UKmUnKoT12encZ4yCa7SigAwMnGCcD5sYz7V0+i/8g1fqa5iun0X/kGr9TQBkT/6+T/eNFE/+vk/ +3jRXcjmNzTP+PMfU1y8//HxL/vn+ddRpn/HmPqa5e4/4+ZMf3z/OuOfxs3j8JGOXAHTnP5ZrxL4w +eJftV/HoltKTDbDdOoJAMp7FehwNpz/tGvWPEurx6B4evNTcY8lPlGMgsTheOO5GfbNfLN7dzX15 +Ncztuklcux9z9ako6f4c6SNW8UQGXlITvbP+fevrmLy7LSo1XGMAYFfMfwr2wzyzHGS2K9wl1hpY +EXfxQB2MV/AYgTgH0rk/EU8dwzggMCMHjtVM6kyrw1Zl5eljkkE4PNAHAWPiFPAHiy4sp0P9jahI +JWdFyYmPG73A7+3SvQdcubHU9HltYriGWKaPzGKncPLHO788fr6V5P8AEZVls45QOUbt6VJZzHwt +8N0uZt0lzqEgXY7Y/dDnA9KAPUvBlh9j0FJmiaOa5kaZvnDArn5CCDxlAvFdDVHRpobrQbKe2iVI +Ht4zGgfOwbRxnuR0/Cr1ABRRRQAV0+i/8g1fqa5iun0X/kGr9TQBkT/6+T/eNFE/+vk/3jRXcjmN +zTP+PMfU1y8//HxLn+8T+tdRpn/HmPqa5LUruCyS6uriRY4YQ7SOx2gAe9cc/jZvH4Tyv41ahZtp +VlpvnobtbjzWjxyF2sOT9SP514k3KjGce9dd4ih1rxHDfeLprWQWDzbA552rnC9hlRwufUiuRkzn +kYNSUdt4Dv8AypXQtjnNeqW+oZjBLDFeBaNfGw1COUfdJwa9Q0/VFkiB3DBoA7Jrvg81SuLvPesr +7eCvUVTur8KnXr3oAq6rZNrl3b6ep4mmRWPouRn9K574kaql5rUNjbsDb2ibFC9AaivfFU9lczra +riRlZBKf4cjGa5uxs7rWNWgtoRJLPcOFBA3H3OPzoA+h/hlHLF8PdLWZGVwsrbHB3YaRiDj0IOfx +zXXVU0u0+waTZ2RcP9mhWJm6btowD7dOmTVugAooooAK6fRf+Qav1NcxXT6L/wAg1fqaAMif/Xyf +7xoon/18n+8aK7kcxt6Z/wAeY+p4rzTxvpGo6/HHpNo5hs7iZjdziTBjQfwhc87vy4r0zTP+PMfU +1y8//HxL/vn+dcc/jZvH4TIn0Cxm8OHRPL/0EQCBRIN+wAYU4PXBwR7gV83+K/C2oeFdTa0vEJjb +JhnH3ZF9vQ+oPNfUfAPt39qy9e0vSdU09k1m2ilt0+YeZnIPqCCKhtRV2Uj5OGQcit/TdaaFFRmI +ArZ1DwDdLPO9hvkt97eSJR8xQHjJ9cda5jUtJutLkVLiMqXGV96zhXpVHyqWo7HULr8ZXmSqV3rq +Mv38iuetrdp3CAhcn7zHH4D1rqLPwcktuxnuCZOPuD7v+NFWrCkryYJXE1zXdKufDVlZ2KD7SBi5 +d48MT9cV6n8NfAjeG4Tql7ldTuE2hFbIhjODt9ycD6Y964Lwz4Pht/FunNf3Aax84MMYBZgcopyD +wWAH4176ORn26g8H8KqnVhNaMT0F4z069j/D6GkoxRVgFFFFABXT6L/yDV+prmK6fRf+Qav1NAGR +P/r5P940UT/6+T/eNFdyOY3NM/48x9TXLz/8fEv++f510+mf8eY+prJl0W7eV2Hl4LEj5q45/Gze +PwmV3rm/Gekalq2kImmyBpIn8w27EATEdPmPAI7ZIHNdt/YV5/0z/wC+qX+xLz/pn9N3Ws5RTVmV +c8igsvF9uqINCkkGMYe5g5H4vxTL/wAH6v4tmt4tRsk0y2hYFmLqzyKc7tuwnGMDrjrXr/8AYd5n +P7vPTlu1B0O87bM+u7oPTpXLTwdKnPnitR82hxtv4J8PW2j/ANlJpsLW2S2XG5gxyNwY87gDgHqO +1Yq/DaK2uXey1m6ghPCRMgkCD05r03+wrvsI8f71J/Yd5jGI/wDvqumUIz+JXEmzi7LwVpVpc290 +4muLiHBDSysV3gYLbc4BzzXRkkkkk89a0f7CvB/zz/76o/sK8/6Z/wDfVOMVH4UBm0Vpf2Fef9M/ +++qP7CvP+mf/AH1VXAzaK0v7CvP+mf8A31R/YV5/0z/76ouBm10+i/8AINX6msr+w7z/AKZ/99Vt +abbva2YikxuBJ4NAGHP/AK+T/eNFE/8Ar35H3jRXcjmLlveSW8YiCKcE8kmpf7XcceUv50UVEoRv +sXFsP7Yf/nkv50n9sP8A88l/OiilyR7Duw/tiT/nkv50f2xJ/wA8l/Oiijkj2C7D+2JP+eS/nR/b +En/PJfzooo5I9guw/tiT/nkv50f2xJ/zyX86KKOSPYLsP7Yk/wCeS/nR/bEn/PJfzooo5I9guw/t +iT/nkv50f2xJ/wA8l/Oiijkj2C7F/th/+eS/nR/a7n/lkv50UUKEewXZmtJudjtHJooorexkf//Z]]></pic> +</pictures> +<category> +<cat>Snacks</cat> +<cat>Cookies & Squares</cat> +</category> +<yield><amount><min>2</min><max>3</max></amount><type>dozen</type></yield> +<preparation-time>00:30</preparation-time> +</krecipes-description> +<krecipes-ingredients> +<ingredient-group name="Dry Ingredients"> +<ingredient> +<name>granulated sugar</name> +<amount><min>0.75</min><max>1</max></amount> +<unit>c.</unit> +</ingredient> +<ingredient> +<name>brown sugar</name> +<amount>1</amount> +<unit>c.</unit> +</ingredient> +<ingredient> +<name>all-purpose flour</name> +<amount>2</amount> +<unit>c.</unit> +</ingredient> +<ingredient> +<name>baking soda</name> +<amount>1</amount> +<unit>tsp.</unit> +</ingredient> +</ingredient-group> +<ingredient-group name="Fat & Liquids"> +<ingredient> +<name>shortening</name> +<amount>1</amount> +<unit>c.</unit> +<prep>softened,at room temperature</prep> +</ingredient> +<ingredient> +<name>peanut butter</name> +<amount>1</amount> +<unit>c.</unit> +</ingredient> +<ingredient> +<name>eggs</name> +<amount>2</amount> +<unit></unit> +</ingredient> +<ingredient> +<name>vanilla extract</name> +<amount>1</amount> +<unit>tsp.</unit> +</ingredient> +</ingredient-group> +<ingredient> + <name>a</name> + <amount>1</amount> + <unit>cup</unit> + <substitutes> + <ingredient> + <name>b</name> + <amount>2</amount> + <unit>cups</unit> + </ingredient> + <ingredient> + <name>c</name> + <amount>3</amount> + <unit>cups</unit> + </ingredient> + </substitutes> +</ingredient> +</krecipes-ingredients> +<krecipes-instructions> +Drop by spoonful on greased cookie sheet. Bake in moderate oven.</krecipes-instructions> +<krecipes-ratings> + <rating> + <rater>George McFry</rater> + <comment>Good enough</comment> + <criterion> + <criteria> + <name>Taste</name> + <stars>5</stars> + </criteria> + </criterion> + </rating> + <rating> + <rater>Me</rater> + <comment>Yuck, don't eat!</comment> + <criterion> + <criteria> + <name>Overall</name> + <stars>2</stars> + </criteria> + <criteria> + <name>Taste</name> + <stars>1.5</stars> + </criteria> + </criterion> + </rating> +</krecipes-ratings> +</krecipes-recipe> +</krecipes> diff --git a/src/tests/mmftest.cpp b/src/tests/mmftest.cpp new file mode 100644 index 0000000..453fcaa --- /dev/null +++ b/src/tests/mmftest.cpp @@ -0,0 +1,139 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include <tdeapplication.h> + +#include <tqstring.h> + +#include <iostream> +using std::cout; +using std::endl; + +#include "mmfimporter.h" +#include "mmfexporter.h" +#include "importertest.h" +#include "exportertest.h" + +int +main(int argc, char *argv[]) +{ + TDEApplication a(argc, argv, "mmftest"); + + printf("Creating MMFImporter.\n"); + MMFImporter importer; + + printf("Parsing mmftest.txt.\n"); + TQStringList files; files << "mmftest.txt"; + importer.parseFiles(files); + + Recipe recipe; + recipe.title = "Cookies_Test"; + recipe.yield.amount = 2; + recipe.yield.type = "servings"; + recipe.categoryList.append( Element("Snacks",1) ); + recipe.categoryList.append( Element("Cookies & Squares",2) ); + recipe.instructions = + "Drop by spoonful on greased cookie sheet. Bake in moderate oven."; + recipe.prepTime = TQTime(0,30); + + Ingredient ing9; + ing9.name = "a"; + ing9.amount = 1; + ing9.amount_offset = 0; + ing9.units.name = "cup"; + IngredientData ing9_1; + ing9_1.name = "b"; + ing9_1.amount = 2; + ing9_1.amount_offset = 0; + ing9_1.units.plural = "cups"; + IngredientData ing9_2; + ing9_2.name = "c"; + ing9_2.amount = 3; + ing9_2.amount_offset = 0; + ing9_2.units.plural = "cups"; + ing9.substitutes.append(ing9_1); + ing9.substitutes.append(ing9_2); + recipe.ingList.append( ing9 ); + + Ingredient ing2; + ing2.name = "c. granulated sugar"; + ing2.amount = 0.75; + ing2.groupID = 0; ing2.group = "Dry Ingredients"; + recipe.ingList.append( ing2 ); + + Ingredient ing; + ing.name = "c. brown sugar"; + ing.amount = 1; + ing.amount_offset = 0; + ing.groupID = 0; ing.group = "Dry Ingredients"; + recipe.ingList.append( ing ); + + Ingredient ing3; + ing3.name = "c. all-purpose flour"; + ing3.amount = 2; + ing3.groupID = 0; ing3.group = "Dry Ingredients"; + recipe.ingList.append( ing3 ); + + Ingredient ing4; + ing4.name = "tsp. baking soda"; + ing4.amount = 1; + ing4.groupID = 0; ing4.group = "Dry Ingredients"; + recipe.ingList.append( ing4 ); + + Ingredient ing8; + ing8.name = "c. shortening"; + ing8.amount = 1; + ing8.prepMethodList.append( Element("softened") ); + ing8.prepMethodList.append( Element("at room temperature") ); + ing8.groupID = 1; ing8.group = "Fat & Liquids"; + recipe.ingList.append( ing8 ); + + Ingredient ing6; + ing6.name = "c. peanut butter"; + ing6.amount = 1; + ing6.groupID = 1; ing6.group = "Fat & Liquids"; + recipe.ingList.append( ing6 ); + + Ingredient ing5; + ing5.name = "eggs"; + ing5.amount = 2; + ing5.groupID = 1; ing5.group = "Fat & Liquids"; + recipe.ingList.append( ing5 ); + + Ingredient ing7; + ing7.name = "tsp. vanilla extract"; + ing7.amount = 1; + ing7.groupID = 1; ing7.group = "Fat & Liquids"; + recipe.ingList.append( ing7 ); + + + check( importer, recipe ); + + RecipeList recipeList; + recipeList.append(recipe); + recipeList.append(recipe); + + printf("Creating MMFExporter.\n"); + MMFExporter exporter("not needed",".mmf"); + check( exporter, recipeList ); + printf("Successfully exported recipes to test.txt.\n"); + + printf("Creating MMFImporter to test exported recipes.\n"); + MMFImporter importer2; + + printf("Parsing test.txt.\n"); + TQStringList files2; files2 << "test.txt"; + importer2.parseFiles(files2); + TQFile::remove("test.txt"); + check( importer2, recipe ); + printf("Recipe export successful.\n"); + + printf("*** MM format importer and exporter passed the tests :-) ***\n"); +} diff --git a/src/tests/mmftest.txt b/src/tests/mmftest.txt new file mode 100644 index 0000000..9d13af3 --- /dev/null +++ b/src/tests/mmftest.txt @@ -0,0 +1,49 @@ +----- Exported by Krecipes vSVN_PRE-0.9 [Meal-Master Export Format] ----- + + Title: Cookies_Test + Categories: Snacks, Cookies & Squares + Servings: 2 + + 1 c a, or + 2 c b, or + 3 c c +------------------------------Dry Ingredients------------------------------- + 3/4 c. granulated sugar + 1 c. brown sugar + 2 c. all-purpose flour + 1 tsp. baking soda +-------------------------------Fat & Liquids-------------------------------- + 1 c. shortening; softened, at + -room temperature + 1 c. peanut butter + 2 eggs + 1 tsp. vanilla extract + + Drop by spoonful on greased cookie sheet. Bake in moderate oven. + +----- + +----- Exported by Krecipes vSVN_PRE-0.9 [Meal-Master Export Format] ----- + + Title: Cookies_Test + Categories: Snacks, Cookies & Squares + Servings: 2 + + 1 c a, or + 2 c b, or + 3 c c +------------------------------Dry Ingredients------------------------------- + 3/4 c. granulated sugar + 1 c. brown sugar + 2 c. all-purpose flour + 1 tsp. baking soda +-------------------------------Fat & Liquids-------------------------------- + 1 c. shortening; softened, at + -room temperature + 1 c. peanut butter + 2 eggs + 1 tsp. vanilla extract + + Drop by spoonful on greased cookie sheet. Bake in moderate oven. + +----- diff --git a/src/tests/mx2test.cpp b/src/tests/mx2test.cpp new file mode 100644 index 0000000..daa379e --- /dev/null +++ b/src/tests/mx2test.cpp @@ -0,0 +1,72 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include <tdeapplication.h> + +#include <tqstring.h> + +#include <iostream> +using std::cout; +using std::endl; + +#include "mx2importer.h" +#include "importertest.h" + +int +main(int argc, char *argv[]) +{ + TDEApplication a(argc, argv, "mx2test"); + + printf("Creating MX2Importer.\n"); + MX2Importer importer; + + printf("Parsing mx2test.txt.\n"); + TQStringList files; files << "mx2test.txt"; + importer.parseFiles(files); + + Recipe recipe; + recipe.title = "Title 1"; + recipe.yield.amount = 2; + recipe.yield.type = "servings"; + recipe.categoryList.append( Element("Category 1") ); + recipe.categoryList.append( Element("Category 2") ); + recipe.instructions = + "Instruction line 1\n" + "Instruction line 2\n" + "Instruction line 3"; + + Ingredient ing; + ing.name = "ingredient 1"; + ing.amount = 1; + ing.units.name = "teaspoon"; + recipe.ingList.append( ing ); + + Ingredient ing2; + ing2.name = "ingredient 2"; + ing2.amount = 3.5; + ing2.units.plural = TQString::null; + recipe.ingList.append( ing2 ); + + Ingredient ing3; + ing3.name = "ingredient 3"; + ing3.amount = 3.5; + ing3.units.plural = "ounces"; + recipe.ingList.append( ing3 ); + + Ingredient ing4; + ing4.name = "ingredient 4"; + ing4.amount = 3.5; + ing4.units.plural = "ounces"; + recipe.ingList.append( ing4 ); + + check( importer, recipe ); + + printf("Done.\n"); +} diff --git a/src/tests/mx2test.txt b/src/tests/mx2test.txt new file mode 100644 index 0000000..e69de29 --- /dev/null +++ b/src/tests/mx2test.txt diff --git a/src/tests/mxptest.cpp b/src/tests/mxptest.cpp new file mode 100644 index 0000000..3c56a6f --- /dev/null +++ b/src/tests/mxptest.cpp @@ -0,0 +1,72 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include <tdeapplication.h> + +#include <tqstring.h> + +#include <iostream> +using std::cout; +using std::endl; + +#include "mxpimporter.h" +#include "importertest.h" + +int +main(int argc, char *argv[]) +{ + TDEApplication a(argc, argv, "mxptest"); + + printf("Creating MXPImporter.\n"); + MXPImporter importer; + + printf("Parsing mxptest.txt.\n"); + TQStringList files; files << "mxptest.txt"; + importer.parseFiles(files); + + Recipe recipe; + recipe.title = "Title 1"; + recipe.yield.amount = 2; + recipe.yield.type = "servings"; + recipe.categoryList.append( Element("Category 1") ); + recipe.categoryList.append( Element("Category 2") ); + recipe.instructions = + "Instruction line 1\n" + "Instruction line 2\n" + "Instruction line 3"; + + Ingredient ing; + ing.name = "ingredient 1"; + ing.amount = 1; + ing.units.name = "teaspoon"; + recipe.ingList.append( ing ); + + Ingredient ing2; + ing2.name = "ingredient 2"; + ing2.amount = 3.5; + ing2.units.plural = TQString::null; + recipe.ingList.append( ing2 ); + + Ingredient ing3; + ing3.name = "ingredient 3"; + ing3.amount = 3.5; + ing3.units.plural = "ounces"; + recipe.ingList.append( ing3 ); + + Ingredient ing4; + ing4.name = "ingredient 4"; + ing4.amount = 3.5; + ing4.units.plural = "ounces"; + recipe.ingList.append( ing4 ); + + check( importer, recipe ); + + printf("Done.\n"); +} diff --git a/src/tests/mxptest.txt b/src/tests/mxptest.txt new file mode 100644 index 0000000..46154c0 --- /dev/null +++ b/src/tests/mxptest.txt @@ -0,0 +1,25 @@ + ----- Exported by Krecipes vSVN_PRE-0.9 [Master Cook Export Format] ----- + + Cookies Test + +Recipe By : Mona Beamer, Colleen Beamer +Serving Size : 2 Preparation Time :0:45 +Categories : Snacks Cookies & Squares + + + Amount Measure Ingredient -- Preparation Method +-------- ------------ -------------------------------- + 3/4 c. granulated sugar + 1 c. brown sugar + 2 c. all-purpose flour + 1 tsp. baking soda + 1 c. shortening -- softened,at room temperature + 1 c. peanut butter + 2 eggs + 1 tsp. vanilla extract + + +Drop by spoonful on greased cookie sheet. Bake in moderate oven. + +Nutr. Assoc. : 0 1374 1021 927 0 1638 1358 797 0 0 0 568 0 532 1611 + diff --git a/src/tests/nyctest.cpp b/src/tests/nyctest.cpp new file mode 100644 index 0000000..724b6a0 --- /dev/null +++ b/src/tests/nyctest.cpp @@ -0,0 +1,100 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include <tdeapplication.h> + +#include <tqstring.h> + +#include <iostream> +using std::cout; +using std::endl; + +#include "nycgenericimporter.h" +#include "importertest.h" + +int +main(int argc, char *argv[]) +{ + TDEApplication a(argc, argv, "nyctest"); + + printf("Creating NYCGenericImporter.\n"); + NYCGenericImporter importer; + + printf("Parsing nyctest.txt.\n"); + TQStringList files; files << "nyctest.txt"; + importer.parseFiles(files); + + Recipe recipe; + recipe.title = "Cookies_Test"; + recipe.yield.amount = 2; + recipe.yield.type = "dozen"; + recipe.categoryList.append( Element("Snacks",1) ); + recipe.categoryList.append( Element("Cookies & Squares",2) ); + recipe.instructions = + "Drop by spoonful on greased cookie sheet. Bake in moderate oven."; + //recipe.prepTime = TQTime(0,30); + + recipe.authorList.append( Element("Colleen Beamer") ); + + Ingredient ing; + ing.name = "granulated sugar"; + ing.amount = 0.75; + ing.units.name = "c."; + recipe.ingList.append( ing ); + + Ingredient ing2; + ing2.name = "brown sugar"; + ing2.amount = 1; + ing2.units.name = "c."; + recipe.ingList.append( ing2 ); + + Ingredient ing3; + ing3.name = "all-purpose flour"; + ing3.amount = 2; + ing3.units.plural = "c."; + recipe.ingList.append( ing3 ); + + Ingredient ing4; + ing4.name = "baking soda"; + ing4.amount = 1; + ing4.units.name = "tsp."; + recipe.ingList.append( ing4 ); + + Ingredient ing8; + ing8.name = "shortening"; + ing8.amount = 1; + ing8.units.name = "c."; + ing8.prepMethodList.append( Element("softened") ); + ing8.prepMethodList.append( Element("at room temperature") ); + recipe.ingList.append( ing8 ); + + Ingredient ing6; + ing6.name = "peanut butter"; + ing6.amount = 1; + ing6.units.name = "c."; + recipe.ingList.append( ing6 ); + + Ingredient ing5; + ing5.name = "eggs"; + ing5.amount = 2; + ing5.units.plural = "whole"; + recipe.ingList.append( ing5 ); + + Ingredient ing7; + ing7.name = "vanilla extract"; + ing7.amount = 1; + ing7.units.name = "tsp."; + + recipe.ingList.append( ing7 ); + + check( importer, recipe ); + + printf("Done.\n"); +} diff --git a/src/tests/nyctest.txt b/src/tests/nyctest.txt new file mode 100644 index 0000000..ad0720b --- /dev/null +++ b/src/tests/nyctest.txt @@ -0,0 +1,57 @@ +@@@@@ Exported by Krecipes vSVN_PRE-0.9 [Now You're Cooking! Export Format] + +Cookies_Test + +Snacks, Cookies & Squares + +3/4 c. granulated sugar +1 c. brown sugar +2 c. all-purpose flour +1 tsp. baking soda +1 c. shortening -- softened,at room temperature +1 c. peanut butter +2 whole eggs +1 tsp. vanilla extract + +Drop by spoonful on greased cookie sheet. Bake in moderate oven. + +NYC Nutrition Analysis (per serving or yield unit): water=70.33 g; calories=394.5; protein=3.14 g; total fat=18 g; carbohydrate=56.94 g; dietary fiber=3.57 g; ash=1.18 g; calcium=14.36 mg; phosphorus=37.48 mg; iron=1.59 mg; sodium=339.7 mg; potassium=111.7 mg; magnesium=10.26 mg; zinc=0.25 mg; copper=0.08 mg; manganese=0.32 mg; vitamin A=160.3 IU; vitamin E=1.43 mg ATE; thiamin=0.2 mg; riboflavin=0.14 mg; niacin=1.6 mg; pantothenic acid=0.13 mg; vitamin B6=0.04 mg; folate=35.9 ug; vitamin C=4.13 mg; saturated fat=5.77 g; monounsaturated fat=6.83 g; polyunsaturated fat=4.41 g; caffeine=1.16 mg; selenium=10.13 ug; refuse=3.34%; %cal as carb:prot:fat=57:3:40; WW Pts=8.7; (complete analysis) + +Contributor: Colleen Beamer + +Yield: 2 dozen + +NYC Nutrilink: N5504^19335,N193^02021,N182^02010,N197^02025 +NYC Nutrilink: N218^02047,N5662^20081,N3906^14355,N1896^09153 +NYC Nutrilink: N1766^09003,N599^04522,U3^18402 + +** Exported from Now You're Cooking! v5.64 ** + +@@@@@ Exported by Krecipes vSVN_PRE-0.9 [Now You're Cooking! Export Format] + +Cookies_Test + +Snacks, Cookies & Squares + +3/4 c. granulated sugar +1 c. brown sugar +2 c. all-purpose flour +1 tsp. baking soda +1 c. shortening -- softened,at room temperature +1 c. peanut butter +2 whole eggs +1 tsp. vanilla extract + +Drop by spoonful on greased cookie sheet. Bake in moderate oven. + +NYC Nutrition Analysis (per serving or yield unit): water=70.33 g; calories=394.5; protein=3.14 g; total fat=18 g; carbohydrate=56.94 g; dietary fiber=3.57 g; ash=1.18 g; calcium=14.36 mg; phosphorus=37.48 mg; iron=1.59 mg; sodium=339.7 mg; potassium=111.7 mg; magnesium=10.26 mg; zinc=0.25 mg; copper=0.08 mg; manganese=0.32 mg; vitamin A=160.3 IU; vitamin E=1.43 mg ATE; thiamin=0.2 mg; riboflavin=0.14 mg; niacin=1.6 mg; pantothenic acid=0.13 mg; vitamin B6=0.04 mg; folate=35.9 ug; vitamin C=4.13 mg; saturated fat=5.77 g; monounsaturated fat=6.83 g; polyunsaturated fat=4.41 g; caffeine=1.16 mg; selenium=10.13 ug; refuse=3.34%; %cal as carb:prot:fat=57:3:40; WW Pts=8.7; (complete analysis) + +Contributor: Colleen Beamer + +Yield: 2 dozen + +NYC Nutrilink: N5504^19335,N193^02021,N182^02010,N197^02025 +NYC Nutrilink: N218^02047,N5662^20081,N3906^14355,N1896^09153 +NYC Nutrilink: N1766^09003,N599^04522,U3^18402 + +** Exported from Now You're Cooking! v5.64 **
\ No newline at end of file diff --git a/src/tests/recipemltest.cpp b/src/tests/recipemltest.cpp new file mode 100644 index 0000000..eb11880 --- /dev/null +++ b/src/tests/recipemltest.cpp @@ -0,0 +1,158 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include <tdeapplication.h> + +#include <tqstring.h> + +#include <iostream> +using std::cout; +using std::endl; + +#include "recipemlimporter.h" +#include "recipemlexporter.h" +#include "importertest.h" +#include "exportertest.h" + +int +main(int argc, char *argv[]) +{ + TDEApplication a(argc, argv, "recipemltest"); + + printf("Creating RecipeMLImporter.\n"); + RecipeMLImporter importer; + + printf("Parsing recipemltest.txt.\n"); + TQStringList files; files << "recipemltest.txt"; + importer.parseFiles(files); + + Recipe recipe; + recipe.title = "Cookies_Test"; + recipe.yield.amount = 2; + recipe.yield.amount_offset = 1; + recipe.yield.type = "dozen"; + recipe.categoryList.append( Element("Snacks",1) ); + recipe.categoryList.append( Element("Cookies & Squares",2) ); + recipe.instructions = + "Drop by spoonful on greased cookie sheet. Bake in moderate oven."; + recipe.prepTime = TQTime(0,30); + + recipe.authorList.append( Element("Colleen Beamer") ); + recipe.authorList.append( Element("Mona Beamer") ); + + Ingredient ing; + ing.name = "granulated sugar"; + ing.amount = 0.75; + ing.amount_offset = 0.25; + ing.units.name = "c."; + ing.groupID = 0; ing.group = "Dry Ingredients"; + recipe.ingList.append( ing ); + + Ingredient ing2; + ing2.name = "brown sugar"; + ing2.amount = 1; + ing2.amount_offset = 0; + ing2.units.name = "c."; + ing2.groupID = 0; ing2.group = "Dry Ingredients"; + recipe.ingList.append( ing2 ); + + Ingredient ing3; + ing3.name = "all-purpose flour"; + ing3.amount = 2; + ing3.units.plural = "c."; + ing3.groupID = 0; ing3.group = "Dry Ingredients"; + recipe.ingList.append( ing3 ); + + Ingredient ing4; + ing4.name = "baking soda"; + ing4.amount = 1; + ing4.amount_offset = 0; + ing4.units.name = "tsp."; + ing4.groupID = 0; ing4.group = "Dry Ingredients"; + recipe.ingList.append( ing4 ); + + Ingredient ing8; + ing8.name = "shortening"; + ing8.amount = 1; + ing8.amount_offset = 0; + ing8.units.name = "c."; + ing8.prepMethodList.append( Element("softened") ); + ing8.prepMethodList.append( Element("at room temperature") ); + ing8.groupID = 1; ing8.group = "Fat & Liquids"; + recipe.ingList.append( ing8 ); + + Ingredient ing6; + ing6.name = "peanut butter"; + ing6.amount = 1; + ing6.amount_offset = 0; + ing6.units.name = "c."; + ing6.groupID = 1; ing6.group = "Fat & Liquids"; + recipe.ingList.append( ing6 ); + + Ingredient ing5; + ing5.name = "eggs"; + ing5.amount = 2; + ing5.amount_offset = 0; + ing5.units.plural = ""; + ing5.groupID = 1; ing5.group = "Fat & Liquids"; + recipe.ingList.append( ing5 ); + + Ingredient ing7; + ing7.name = "vanilla extract"; + ing7.amount = 1; + ing7.amount_offset = 0; + ing7.units.name = "tsp."; + ing7.groupID = 1; ing7.group = "Fat & Liquids"; + recipe.ingList.append( ing7 ); + + Ingredient ing9; + ing9.name = "a"; + ing9.amount = 1; + ing9.amount_offset = 0; + ing9.units.name = "cup"; + IngredientData ing9_1; + ing9_1.name = "b"; + ing9_1.amount = 2; + ing9_1.amount_offset = 0; + ing9_1.units.plural = "cups"; + IngredientData ing9_2; + ing9_2.name = "c"; + ing9_2.amount = 3; + ing9_2.amount_offset = 0; + ing9_2.units.plural = "cups"; + ing9.substitutes.append(ing9_1); + ing9.substitutes.append(ing9_2); + recipe.ingList.append( ing9 ); + + check( importer, recipe ); + + RecipeList recipeList; + recipeList.append(recipe); + recipeList.append(recipe); + + printf("Creating RecipeMLExporter.\n"); + RecipeMLExporter exporter("not needed",".mmf"); + check( exporter, recipeList ); + printf("Successfully exported recipes to test.txt.\n"); + + printf("Creating RecipeMLImporter to test exported recipes.\n"); + RecipeMLImporter importer2; + + printf("Parsing test.txt.\n"); + TQStringList files2; files2 << "test.txt"; + importer2.parseFiles(files2); + TQFile::remove("test.txt"); + check( importer2, recipe ); + printf("Recipe export successful.\n"); + + printf("*** RecipeML importer and exporter passed the tests :-) ***\n"); + + printf("Done.\n"); +} diff --git a/src/tests/recipemltest.txt b/src/tests/recipemltest.txt new file mode 100644 index 0000000..330f757 --- /dev/null +++ b/src/tests/recipemltest.txt @@ -0,0 +1,249 @@ +<?xml version="1.0" encoding="UTF-8" ?> +<!DOCTYPE recipeml PUBLIC "-//FormatData//DTD RecipeML 0.5//EN" "http://www.formatdata.com/recipeml/recipeml.dtd"><recipeml version="0.5" generator="Krecipes vSVN_PRE-0.9"> +<recipe> + <head> + <title>Cookies_Test</title> + <source> + <srcitem>Colleen Beamer</srcitem> + <srcitem>Mona Beamer</srcitem> + </source> + <categories> + <cat>Snacks</cat> + <cat>Cookies & Squares</cat> + </categories> + <yield> + <range> + <q1>2</q1> + <q2>3</q2> + </range> + <unit>dozen</unit> + </yield> + <preptime type="Total" > + <time> + <qty>30</qty> + <timeunit>minutes</timeunit> + </time> + </preptime> + </head> + <ingredients> + <ing-div> + <title>Dry Ingredients</title> + <ing> + <amt> + <qty> + <range> + <q1>0.75</q1> + <q2>1</q2> + </range> + </qty> + <unit>c.</unit> + </amt> + <item>granulated sugar</item> + </ing> + <ing> + <amt> + <qty>1</qty> + <unit>c.</unit> + </amt> + <item>brown sugar</item> + </ing> + <ing> + <amt> + <qty>2</qty> + <unit>c.</unit> + </amt> + <item>all-purpose flour</item> + </ing> + <ing> + <amt> + <qty>1</qty> + <unit>tsp.</unit> + </amt> + <item>baking soda</item> + </ing> + </ing-div> + <ing-div> + <title>Fat & Liquids</title> + <ing> + <amt> + <qty>1</qty> + <unit>c.</unit> + </amt> + <item>shortening</item> + <prep>softened,at room temperature</prep> + </ing> + <ing> + <amt> + <qty>1</qty> + <unit>c.</unit> + </amt> + <item>peanut butter</item> + </ing> + <ing> + <amt> + <qty>2</qty> + <unit></unit> + </amt> + <item>eggs</item> + </ing> + <ing> + <amt> + <qty>1</qty> + <unit>tsp.</unit> + </amt> + <item>vanilla extract</item> + </ing> + </ing-div> + <ing> + <amt> + <qty>1</qty> + <unit>cup</unit> + </amt> + <item>a</item> + <prep></prep> + <alt-ing> + <amt> + <qty>2</qty> + <unit>cups</unit> + </amt> + <item>b</item> + <prep></prep> + </alt-ing> + <alt-ing> + <amt> + <qty>3</qty> + <unit>cups</unit> + </amt> + <item>c</item> + <prep></prep> + </alt-ing> + </ing> + </ingredients> + <directions> + <step>Drop by spoonful on greased cookie sheet. Bake in moderate oven.</step> + </directions> +</recipe> +<recipe> + <head> + <title>Cookies_Test</title> + <source> + <srcitem>Colleen Beamer</srcitem> + <srcitem>Mona Beamer</srcitem> + </source> + <categories> + <cat>Snacks</cat> + <cat>Cookies & Squares</cat> + </categories> + <yield> + <range> + <q1>2</q1> + <q2>3</q2> + </range> + <unit>dozen</unit> + </yield> + <preptime type="Total" > + <time> + <qty>30</qty> + <timeunit>minutes</timeunit> + </time> + </preptime> + </head> + <ingredients> + <ing-div> + <title>Dry Ingredients</title> + <ing> + <amt> + <qty> + <range> + <q1>0.75</q1> + <q2>1</q2> + </range> + </qty> + <unit>c.</unit> + </amt> + <item>granulated sugar</item> + </ing> + <ing> + <amt> + <qty>1</qty> + <unit>c.</unit> + </amt> + <item>brown sugar</item> + </ing> + <ing> + <amt> + <qty>2</qty> + <unit>c.</unit> + </amt> + <item>all-purpose flour</item> + </ing> + <ing> + <amt> + <qty>1</qty> + <unit>tsp.</unit> + </amt> + <item>baking soda</item> + </ing> + </ing-div> + <ing-div> + <title>Fat & Liquids</title> + <ing> + <amt> + <qty>1</qty> + <unit>c.</unit> + </amt> + <item>shortening</item> + <prep>softened,at room temperature</prep> + </ing> + <ing> + <amt> + <qty>1</qty> + <unit>c.</unit> + </amt> + <item>peanut butter</item> + </ing> + <ing> + <amt> + <qty>2</qty> + <unit></unit> + </amt> + <item>eggs</item> + </ing> + <ing> + <amt> + <qty>1</qty> + <unit>tsp.</unit> + </amt> + <item>vanilla extract</item> + </ing> + </ing-div> + <ing> + <amt> + <qty>1</qty> + <unit>cup</unit> + </amt> + <item>a</item> + <prep></prep> + <alt-ing> + <amt> + <qty>2</qty> + <unit>cups</unit> + </amt> + <item>b</item> + <prep></prep> + </alt-ing> + <alt-ing> + <amt> + <qty>3</qty> + <unit>cups</unit> + </amt> + <item>c</item> + <prep></prep> + </alt-ing> + </ing> + </ingredients> + <directions> + <step>Drop by spoonful on greased cookie sheet. Bake in moderate oven.</step> + </directions> +</recipe> +</recipeml>
\ No newline at end of file diff --git a/src/tests/rezkonvtest.cpp b/src/tests/rezkonvtest.cpp new file mode 100644 index 0000000..d9b8022 --- /dev/null +++ b/src/tests/rezkonvtest.cpp @@ -0,0 +1,155 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include <tdeapplication.h> + +#include <tqstring.h> + +#include <iostream> +using std::cout; +using std::endl; + +#include "rezkonvimporter.h" +#include "rezkonvexporter.h" +#include "importertest.h" +#include "exportertest.h" + +int +main(int argc, char *argv[]) +{ + TDEApplication a(argc, argv, "rezkonvtest"); + + printf("Creating RezkonvImporter.\n"); + RezkonvImporter importer; + + printf("Parsing rezkonvtest.txt.\n"); + TQStringList files; files << "rezkonvtest.txt"; + importer.parseFiles(files); + + Recipe recipe; + recipe.title = "Cookies_Test"; + recipe.yield.amount = 2; + recipe.yield.amount_offset = 1; + recipe.yield.type = "dozen"; + recipe.categoryList.append( Element("Snacks") ); + recipe.categoryList.append( Element("Cookies & Squares") ); + recipe.instructions = + "\n\nDrop by spoonful on greased cookie sheet. Bake in moderate oven."; + + recipe.authorList.append( Element("Mona Beamer") ); + recipe.authorList.append( Element("Colleen Beamer") ); + + Ingredient ing9; + ing9.name = "a"; + ing9.amount = 1; + ing9.amount_offset = 0; + ing9.units.name = "cup"; + IngredientData ing9_1; + ing9_1.name = "b"; + ing9_1.amount = 2; + ing9_1.amount_offset = 0; + ing9_1.units.plural = "cups"; + IngredientData ing9_2; + ing9_2.name = "c"; + ing9_2.amount = 3; + ing9_2.amount_offset = 0; + ing9_2.units.plural = "cups"; + ing9.substitutes.append(ing9_1); + ing9.substitutes.append(ing9_2); + recipe.ingList.append( ing9 ); + + Ingredient ing; + ing.name = "granulated sugar"; + ing.amount = 0.75; + ing.amount_offset = 0.25; + ing.units.name = "c."; + ing.groupID = 0; ing.group = "Dry Ingredients"; + recipe.ingList.append( ing ); + + Ingredient ing2; + ing2.name = "brown sugar"; + ing2.amount = 1; + ing2.amount_offset = 0; + ing2.units.name = "c."; + ing2.groupID = 0; ing2.group = "Dry Ingredients"; + recipe.ingList.append( ing2 ); + + Ingredient ing3; + ing3.name = "all-purpose flour"; + ing3.amount = 2; + ing3.units.plural = "c."; + ing3.groupID = 0; ing3.group = "Dry Ingredients"; + recipe.ingList.append( ing3 ); + + Ingredient ing4; + ing4.name = "baking soda"; + ing4.amount = 1; + ing4.amount_offset = 0; + ing4.units.name = "tsp."; + ing4.groupID = 0; ing4.group = "Dry Ingredients"; + recipe.ingList.append( ing4 ); + + Ingredient ing8; + ing8.name = "shortening"; + ing8.amount = 1; + ing8.amount_offset = 0; + ing8.units.name = "c."; + ing8.prepMethodList.append( Element("softened") ); + ing8.prepMethodList.append( Element("at room temperature") ); + ing8.groupID = 1; ing8.group = "Fat & Liquids"; + recipe.ingList.append( ing8 ); + + Ingredient ing6; + ing6.name = "peanut butter"; + ing6.amount = 1; + ing6.amount_offset = 0; + ing6.units.name = "c."; + ing6.groupID = 1; ing6.group = "Fat & Liquids"; + recipe.ingList.append( ing6 ); + + Ingredient ing5; + ing5.name = "eggs"; + ing5.amount = 2; + ing5.amount_offset = 0; + ing5.units.plural = ""; + ing5.groupID = 1; ing5.group = "Fat & Liquids"; + recipe.ingList.append( ing5 ); + + Ingredient ing7; + ing7.name = "vanilla extract"; + ing7.amount = 1; + ing7.amount_offset = 0; + ing7.units.name = "tsp."; + ing7.groupID = 1; ing7.group = "Fat & Liquids"; + recipe.ingList.append( ing7 ); + + check( importer, recipe ); + + RecipeList recipeList; + recipeList.append(recipe); + recipeList.append(recipe); + + printf("Creating RezkonvExporter.\n"); + RezkonvExporter exporter("not needed",".rk"); + check( exporter, recipeList ); + printf("Successfully exported recipes to test.txt.\n"); + + printf("Creating RezkonvImporter to test exported recipes.\n"); + RezkonvImporter importer2; + + printf("Parsing test.txt.\n"); + TQStringList files2; files2 << "test.txt"; + importer2.parseFiles(files2); + TQFile::remove("test.txt"); + check( importer2, recipe ); + printf("Recipe export successful.\n"); + + printf("Done.\n"); +} diff --git a/src/tests/rezkonvtest.txt b/src/tests/rezkonvtest.txt new file mode 100644 index 0000000..38e4ac2 --- /dev/null +++ b/src/tests/rezkonvtest.txt @@ -0,0 +1,55 @@ +========== REZKONV-Rezept - RezkonvSuite v0.97.1 + + Titel: Cookies_Test +Kategorien: Snacks, Cookies & Squares + Menge: 2-3 dozen + + 1 cup a, or + 2 cups b, or + 3 cups c +========================== Dry Ingredients ========================== + 3/4-1 c. granulated sugar + 1 c. brown sugar + 2 c. all-purpose flour + 1 tsp. baking soda + +=========================== Fat & Liquids =========================== + 1 c. shortening, softened,at room temperature + 1 c. peanut butter + 2 eggs + 1 tsp. vanilla extract + +============================== QUELLE ============================== + Mona Beamer + -- Colleen Beamer + +Drop by spoonful on greased cookie sheet. Bake in moderate oven. +===== + +========== REZKONV-Rezept - RezkonvSuite v0.97.1 + + Titel: Cookies_Test +Kategorien: Snacks, Cookies & Squares + Menge: 2-3 dozen + + 1 cup a, or + 2 cups b, or + 3 cups c +========================== Dry Ingredients ========================== + 3/4-1 c. granulated sugar + 1 c. brown sugar + 2 c. all-purpose flour + 1 tsp. baking soda + +=========================== Fat & Liquids =========================== + 1 c. shortening, softened,at room temperature + 1 c. peanut butter + 2 eggs + 1 tsp. vanilla extract + +============================== QUELLE ============================== + Mona Beamer + -- Colleen Beamer + +Drop by spoonful on greased cookie sheet. Bake in moderate oven. +===== diff --git a/src/tests/test_photo.jpg b/src/tests/test_photo.jpg Binary files differnew file mode 100644 index 0000000..3ebe035 --- /dev/null +++ b/src/tests/test_photo.jpg diff --git a/src/widgets/Makefile.am b/src/widgets/Makefile.am new file mode 100644 index 0000000..b92e241 --- /dev/null +++ b/src/widgets/Makefile.am @@ -0,0 +1,28 @@ +## Makefile.am for krecipes + +# this is the program that gets installed. it's name is used for all +# of the other Makefile.am variables + +# set the include path for X, tqt and TDE +INCLUDES = -I$(srcdir)/.. $(all_includes) + +noinst_LTLIBRARIES=libkrecipeswidgets.la + +libkrecipeswidgets_la_SOURCES= \ + krelistview.cpp kremenu.cpp \ + paneldeco.cpp ingredientlistview.cpp unitlistview.cpp \ + propertylistview.cpp prepmethodlistview.cpp categorylistview.cpp \ + authorlistview.cpp recipelistview.cpp categorycombobox.cpp \ + kretextedit.cpp dblistviewbase.cpp \ + conversiontable.cpp fractioninput.cpp ingredientcombobox.cpp \ + headercombobox.cpp prepmethodcombobox.cpp \ + inglistviewitem.cpp kdateedit.cpp kdatepickerpopup.cpp \ + headerlistview.cpp ratingwidget.cpp kwidgetlistbox.cpp \ + ratingdisplaywidget.ui criteriacombobox.cpp ingredientinputwidget.cpp \ + unitcombobox.cpp amountunitinput.cpp weightinput.cpp + + +libkrecipeswidgets_la_METASOURCES=AUTO + +#the library search path. +libkrecipeswidgets_la_LDFLAGS = $(KDE_RPATH) $(all_libraries) diff --git a/src/widgets/amountunitinput.cpp b/src/widgets/amountunitinput.cpp new file mode 100644 index 0000000..699f76d --- /dev/null +++ b/src/widgets/amountunitinput.cpp @@ -0,0 +1,68 @@ +/*************************************************************************** +* Copyright (C) 2006 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "amountunitinput.h" + +#include <tqheader.h> +#include <tqlistview.h> + +#include "fractioninput.h" +#include "unitcombobox.h" +#include "backends/recipedb.h" +#include "datablocks/mixednumber.h" + +AmountUnitInput::AmountUnitInput( TQWidget *parent, RecipeDB *database, Unit::Type type, MixedNumber::Format format ) : TQHBox(parent), + m_item(0), m_database(database) +{ + amountInput = new FractionInput(this,format); + unitBox = new UnitComboBox(this,database,type); + unitBox->reload(); + + connect( amountInput, TQ_SIGNAL(valueChanged(const MixedNumber &)), TQ_SLOT(emitValueChanged()) ); + connect( unitBox, TQ_SIGNAL(activated(int)), TQ_SLOT(emitValueChanged()) ); + connect( amountInput, TQ_SIGNAL(returnPressed()), TQ_SIGNAL(doneEditing()) ); +} + +void AmountUnitInput::emitValueChanged() +{ + emit valueChanged( amount(), unit() ); +} + +void AmountUnitInput::setAmount( const MixedNumber &amount ) +{ + amountInput->disconnect( this ); + if ( amount.toDouble() < 0 ) + amountInput->clear(); + else + amountInput->setValue( amount, 0 ); + connect( amountInput, TQ_SIGNAL(valueChanged(const MixedNumber &)), TQ_SLOT(emitValueChanged()) ); +} + +void AmountUnitInput::setUnit( const Unit &unit ) +{ + if ( unit.id == -1 ) + unitBox->setCurrentItem(0); + else + unitBox->setSelected( unit.id ); + +} + +MixedNumber AmountUnitInput::amount() const +{ + return amountInput->value(); +} + +Unit AmountUnitInput::unit() const +{ + //TODO Potential for optimization here... avoid the database call + return m_database->unitName( unitBox->id( unitBox->currentItem() ) ); +} + +#include "amountunitinput.moc" diff --git a/src/widgets/amountunitinput.h b/src/widgets/amountunitinput.h new file mode 100644 index 0000000..911e67c --- /dev/null +++ b/src/widgets/amountunitinput.h @@ -0,0 +1,59 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* * +* Copyright (C) 2006 Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef AMOUNTUNITINPUT_H +#define AMOUNTUNITINPUT_H + +#include <tqhbox.h> + +#include "datablocks/unit.h" +#include "datablocks/mixednumber.h" + +class TQListViewItem; + +class RecipeDB; +class FractionInput; +class UnitComboBox; + +class AmountUnitInput : public TQHBox +{ +TQ_OBJECT + +public: + AmountUnitInput( TQWidget *parent, RecipeDB *database, Unit::Type type = Unit::All, MixedNumber::Format f = MixedNumber::MixedNumberFormat ); + + void setAmount( const MixedNumber &amount ); + void setUnit( const Unit &unit ); + + MixedNumber amount() const; + Unit unit() const; + TQListViewItem *item() const { return m_item; } + void setItem( TQListViewItem *it ){ m_item = it; } + + void insertIntoListview( TQListViewItem *it, int col ); + +public slots: + void emitValueChanged(); + +signals: + void valueChanged( const MixedNumber &, const Unit &unit ); + void doneEditing(); + +private: + FractionInput *amountInput; + UnitComboBox *unitBox; + + TQListViewItem *m_item; + + RecipeDB *m_database; +}; +#endif diff --git a/src/widgets/authorlistview.cpp b/src/widgets/authorlistview.cpp new file mode 100644 index 0000000..7ef696d --- /dev/null +++ b/src/widgets/authorlistview.cpp @@ -0,0 +1,275 @@ +/*************************************************************************** +* Copyright (C) 2004 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "authorlistview.h" + +#include <tdemessagebox.h> +#include <tdeconfig.h> +#include <tdelocale.h> +#include <tdeglobal.h> +#include <kiconloader.h> +#include <tdepopupmenu.h> + +#include "backends/recipedb.h" +#include "dialogs/createelementdialog.h" +#include "dialogs/dependanciesdialog.h" + +AuthorListView::AuthorListView( TQWidget *parent, RecipeDB *db ) : DBListViewBase( parent, db, db->authorCount() ) +{ + setAllColumnsShowFocus( true ); + setDefaultRenameAction( TQListView::Reject ); +} + +void AuthorListView::init() +{ + connect( database, TQ_SIGNAL( authorCreated( const Element & ) ), TQ_SLOT( checkCreateAuthor( const Element & ) ) ); + connect( database, TQ_SIGNAL( authorRemoved( int ) ), TQ_SLOT( removeAuthor( int ) ) ); +} + +void AuthorListView::load( int limit, int offset ) +{ + ElementList authorList; + database->loadAuthors( &authorList, limit, offset ); + + setTotalItems(authorList.count()); + + for ( ElementList::const_iterator ing_it = authorList.begin(); ing_it != authorList.end(); ++ing_it ) + createAuthor( *ing_it ); +} + +void AuthorListView::checkCreateAuthor( const Element &el ) +{ + if ( handleElement(el.name) ) { //only create this author if the base class okays it + createAuthor(el); + } +} + + +StdAuthorListView::StdAuthorListView( TQWidget *parent, RecipeDB *db, bool editable ) : AuthorListView( parent, db ) +{ + addColumn( i18n( "Author" ) ); + + TDEConfig * config = TDEGlobal::config(); + config->setGroup( "Advanced" ); + bool show_id = config->readBoolEntry( "ShowID", false ); + addColumn( i18n( "Id" ), show_id ? -1 : 0 ); + + if ( editable ) { + setRenameable( 0, true ); + + TDEIconLoader *il = new TDEIconLoader; + + kpop = new TDEPopupMenu( this ); + kpop->insertItem( il->loadIcon( "document-new", TDEIcon::NoGroup, 16 ), i18n( "&Create" ), this, TQ_SLOT( createNew() ), CTRL + Key_N ); + kpop->insertItem( il->loadIcon( "edit-delete", TDEIcon::NoGroup, 16 ), i18n( "&Delete" ), this, TQ_SLOT( remove + () ), Key_Delete ); + kpop->insertItem( il->loadIcon( "edit", TDEIcon::NoGroup, 16 ), i18n( "&Rename" ), this, TQ_SLOT( rename() ), CTRL + Key_R ); + kpop->polish(); + + delete il; + + connect( this, TQ_SIGNAL( contextMenu( TDEListView *, TQListViewItem *, const TQPoint & ) ), TQ_SLOT( showPopup( TDEListView *, TQListViewItem *, const TQPoint & ) ) ); + connect( this, TQ_SIGNAL( doubleClicked( TQListViewItem* ) ), this, TQ_SLOT( modAuthor( TQListViewItem* ) ) ); + connect( this, TQ_SIGNAL( itemRenamed( TQListViewItem* ) ), this, TQ_SLOT( saveAuthor( TQListViewItem* ) ) ); + } +} + +void StdAuthorListView::showPopup( TDEListView * /*l*/, TQListViewItem *i, const TQPoint &p ) +{ + if ( i ) + kpop->exec( p ); +} + +void StdAuthorListView::createNew() +{ + CreateElementDialog * elementDialog = new CreateElementDialog( this, i18n( "New Author" ) ); + + if ( elementDialog->exec() == TQDialog::Accepted ) { + TQString result = elementDialog->newElementName(); + + //check bounds first + if ( checkBounds( result ) ) + database->createNewAuthor( result ); // Create the new author in the database + } +} + +void StdAuthorListView::remove + () +{ + TQListViewItem * item = currentItem(); + + if ( item ) { + int id = item->text( 1 ).toInt(); + + ElementList recipeDependancies; + database->findUseOfAuthorInRecipes( &recipeDependancies, id ); + + if ( recipeDependancies.isEmpty() ) { + switch ( KMessageBox::warningContinueCancel( this, i18n( "Are you sure you want to delete this author?" ) ) ) { + case KMessageBox::Continue: + database->removeAuthor( id ); + break; + } + return; + } + else { // need warning! + ListInfo info; + info.list = recipeDependancies; + info.name = i18n("Recipes"); + + DependanciesDialog warnDialog( this, info, false ); + if ( warnDialog.exec() == TQDialog::Accepted ) + database->removeAuthor( id ); + } + } +} + +void StdAuthorListView::rename() +{ + TQListViewItem * item = currentItem(); + + if ( item ) + AuthorListView::rename( item, 0 ); +} + +void StdAuthorListView::createAuthor( const Element &author ) +{ + createElement(new TQListViewItem( this, author.name, TQString::number( author.id ) )); +} + +void StdAuthorListView::removeAuthor( int id ) +{ + TQListViewItem * item = findItem( TQString::number( id ), 1 ); + removeElement(item); +} + +void StdAuthorListView::modAuthor( TQListViewItem* i ) +{ + if ( i ) + AuthorListView::rename( i, 0 ); +} + +void StdAuthorListView::saveAuthor( TQListViewItem* i ) +{ + if ( !checkBounds( i->text( 0 ) ) ) { + reload(ForceReload); //reset the changed text + return ; + } + + int existing_id = database->findExistingAuthorByName( i->text( 0 ) ); + int author_id = i->text( 1 ).toInt(); + if ( existing_id != -1 && existing_id != author_id ) //category already exists with this label... merge the two + { + switch ( KMessageBox::warningContinueCancel( this, i18n( "This author already exists. Continuing will merge these two authors into one. Are you sure?" ) ) ) + { + case KMessageBox::Continue: { + database->mergeAuthors( existing_id, author_id ); + break; + } + default: + reload(ForceReload); + break; + } + } + else { + database->modAuthor( ( i->text( 1 ) ).toInt(), i->text( 0 ) ); + } +} + +bool StdAuthorListView::checkBounds( const TQString &name ) +{ + if ( name.length() > uint(database->maxAuthorNameLength()) ) { + KMessageBox::error( this, TQString( i18n( "Author name cannot be longer than %1 characters." ) ).arg( database->maxAuthorNameLength() ) ); + return false; + } + + return true; +} + + +AuthorCheckListItem::AuthorCheckListItem( AuthorCheckListView* qlv, const Element &author ) : TQCheckListItem( qlv, TQString::null, TQCheckListItem::CheckBox ), + authorStored(author), + m_listview(qlv) +{ +} + +AuthorCheckListItem::AuthorCheckListItem( AuthorCheckListView* qlv, TQListViewItem *after, const Element &author ) : TQCheckListItem( qlv, after, TQString::null, TQCheckListItem::CheckBox ), + authorStored(author), + m_listview(qlv) +{ +} + +Element AuthorCheckListItem::author() const +{ + return authorStored; +} + +TQString AuthorCheckListItem::text( int column ) const +{ + switch ( column ) { + case 0: + return ( authorStored.name ); + case 1: + return ( TQString::number( authorStored.id ) ); + default: + return TQString::null; + } +} + +void AuthorCheckListItem::stateChange( bool on ) +{ + m_listview->stateChange(this,on); +} + + +AuthorCheckListView::AuthorCheckListView( TQWidget *parent, RecipeDB *db ) : AuthorListView( parent, db ) +{ + addColumn( i18n( "Author" ) ); + + TDEConfig *config = TDEGlobal::config(); + config->setGroup( "Advanced" ); + bool show_id = config->readBoolEntry( "ShowID", false ); + addColumn( i18n( "Id" ), show_id ? -1 : 0 ); +} + +void AuthorCheckListView::createAuthor( const Element &author ) +{ + createElement(new AuthorCheckListItem( this, author )); +} + +void AuthorCheckListView::removeAuthor( int id ) +{ + TQListViewItem * item = findItem( TQString::number( id ), 1 ); + removeElement(item); +} + +void AuthorCheckListView::load( int limit, int offset ) +{ + AuthorListView::load(limit,offset); + + for ( TQValueList<Element>::const_iterator author_it = m_selections.begin(); author_it != m_selections.end(); ++author_it ) { + TQCheckListItem * item = ( TQCheckListItem* ) findItem( TQString::number( (*author_it).id ), 1 ); + if ( item ) { + item->setOn(true); + } + } +} + +void AuthorCheckListView::stateChange(AuthorCheckListItem *it,bool on) +{ + if ( !reloading() ) { + if ( on ) + m_selections.append(it->author()); + else + m_selections.remove(it->author()); + } +} + +#include "authorlistview.moc" diff --git a/src/widgets/authorlistview.h b/src/widgets/authorlistview.h new file mode 100644 index 0000000..a1cdc49 --- /dev/null +++ b/src/widgets/authorlistview.h @@ -0,0 +1,107 @@ +/*************************************************************************** +* Copyright (C) 2004 by * +* Jason Kivlighn ([email protected]) * +* Unai Garro ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef AUTHORLISTVIEW_H +#define AUTHORLISTVIEW_H + +#include "dblistviewbase.h" +#include "datablocks/element.h" + +class RecipeDB; +class TDEPopupMenu; + +class AuthorCheckListView; + +class AuthorCheckListItem: public TQCheckListItem +{ +public: + AuthorCheckListItem( AuthorCheckListView* qlv, const Element &author ); + AuthorCheckListItem( AuthorCheckListView* qlv, TQListViewItem *after, const Element &author ); + + Element author() const; + + virtual TQString text( int column ) const; + +protected: + virtual void stateChange( bool on ); + +private: + Element authorStored; + AuthorCheckListView *m_listview; +}; + + +class AuthorListView : public DBListViewBase +{ + TQ_OBJECT + +public: + AuthorListView( TQWidget *parent, RecipeDB *db ); + +protected slots: + void checkCreateAuthor( const Element &el ); + virtual void createAuthor( const Element & ) = 0; + virtual void removeAuthor( int ) = 0; + virtual void load( int curr_limit, int curr_offset ); + +protected: + virtual void init(); +}; + +class StdAuthorListView : public AuthorListView +{ + TQ_OBJECT + +public: + StdAuthorListView( TQWidget *parent, RecipeDB *db, bool editable = false ); + +protected: + virtual void createAuthor( const Element & ); + virtual void removeAuthor( int ); + +private slots: + void showPopup( TDEListView *, TQListViewItem *, const TQPoint & ); + + void createNew(); + void remove + (); + void rename(); + + void modAuthor( TQListViewItem* i ); + void saveAuthor( TQListViewItem* i ); + +private: + bool checkBounds( const TQString &name ); + + TDEPopupMenu *kpop; +}; + + +class AuthorCheckListView : public AuthorListView +{ +public: + AuthorCheckListView( TQWidget *parent, RecipeDB *db ); + + virtual void stateChange(AuthorCheckListItem *,bool); + + TQValueList<Element> selections() const{ return m_selections; } + +protected: + virtual void createAuthor( const Element &ing ); + virtual void removeAuthor( int ); + + virtual void load( int limit, int offset ); + +private: + TQValueList<Element> m_selections; +}; + +#endif //AUTHORLISTVIEW_H diff --git a/src/widgets/categorycombobox.cpp b/src/widgets/categorycombobox.cpp new file mode 100644 index 0000000..7935cbb --- /dev/null +++ b/src/widgets/categorycombobox.cpp @@ -0,0 +1,182 @@ +/*************************************************************************** +* Copyright (C) 2003-2004 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "categorycombobox.h" + +#include <tqlistbox.h> + +#include <tdelocale.h> +#include <tdeconfig.h> +#include <tdeglobal.h> + +#include "backends/recipedb.h" +#include "backends/progressinterface.h" +#include "datablocks/elementlist.h" +#include "datablocks/categorytree.h" + +CategoryComboBox::CategoryComboBox( TQWidget *parent, RecipeDB *db ) : KComboBox( parent ), + database( db ), + m_offset(0) +{ + connect( database, TQ_SIGNAL( categoryCreated( const Element &, int ) ), TQ_SLOT( createCategory( const Element &, int ) ) ); + connect( database, TQ_SIGNAL( categoryRemoved( int ) ), TQ_SLOT( removeCategory( int ) ) ); + connect( database, TQ_SIGNAL( categoryModified( const Element & ) ), TQ_SLOT( modifyCategory( const Element & ) ) ); + connect( database, TQ_SIGNAL( categoriesMerged( int, int ) ), TQ_SLOT( mergeCategories( int, int ) ) ); + + // Insert default "All Categories" (row 0, which will be translated to -1 as category in the filtering process) + // the rest of the items are loaded when needed in order to significantly speed up startup + insertItem( i18n( "All Categories" ) ); +} + +void CategoryComboBox::popup() +{ + if ( count() == 1 ) + reload(); + KComboBox::popup(); +} + +void CategoryComboBox::reload() +{ + TQString remember_cat_filter = currentText(); + + TDEConfig * config = TDEGlobal::config();config->setGroup( "Performance" ); + int limit = config->readNumEntry( "CategoryLimit", -1 ); + + //ProgressInterface pi(this); + //pi.listenOn(database); + + CategoryTree categoryList; + database->loadCategories( &categoryList, limit, m_offset, -1 ); + + clear(); + categoryComboRows.clear(); + + // Insert default "All Categories" (row 0, which will be translated to -1 as category in the filtering process) + insertItem( i18n( "All Categories" ) ); + + //Now load the categories + int row = 1; + loadCategories(&categoryList,row); + + if ( listBox() ->findItem( remember_cat_filter, TQt::ExactMatch ) ) { + setCurrentText( remember_cat_filter ); + } +} + +void CategoryComboBox::loadCategories( CategoryTree *categoryTree, int &row ) +{ + for ( CategoryTree * child_it = categoryTree->firstChild(); child_it; child_it = child_it->nextSibling() ) { + insertItem( child_it->category.name ); + categoryComboRows.insert( row, child_it->category.id ); // store category id's in the combobox position to obtain the category id later + row++; + loadCategories( child_it, row ); + } +} + +void CategoryComboBox::loadNextGroup() +{ + TDEConfig * config = TDEGlobal::config();config->setGroup( "Performance" ); + int limit = config->readNumEntry( "CategoryLimit", -1 ); + + m_offset += limit; + + reload(); +} + +void CategoryComboBox::loadPrevGroup() +{ + TDEConfig * config = TDEGlobal::config();config->setGroup( "Performance" ); + int limit = config->readNumEntry( "CategoryLimit", -1 ); + + m_offset -= limit; + + reload(); +} + +int CategoryComboBox::id( int row ) +{ + if ( row ) + return categoryComboRows[ row ]; + else + return -1; // No category filtering +} + +void CategoryComboBox::createCategory( const Element &element, int /*parent_id*/ ) +{ + int row = findInsertionPoint( element.name ); + + insertItem( element.name, row ); + + //now update the map by pushing everything after this item down + TQMap<int, int> new_map; + for ( TQMap<int, int>::iterator it = categoryComboRows.begin(); it != categoryComboRows.end(); ++it ) { + if ( it.key() >= row ) { + new_map.insert( it.key() + 1, it.data() ); + } + else + new_map.insert( it.key(), it.data() ); + } + categoryComboRows = new_map; + categoryComboRows.insert( row, element.id ); +} + +void CategoryComboBox::removeCategory( int id ) +{ + int row = -1; + for ( TQMap<int, int>::iterator it = categoryComboRows.begin(); it != categoryComboRows.end(); ++it ) { + if ( it.data() == id ) { + row = it.key(); + removeItem( row ); + categoryComboRows.remove( it ); + break; + } + } + + if ( row == -1 ) + return ; + + //now update the map by pushing everything after this item up + TQMap<int, int> new_map; + for ( TQMap<int, int>::iterator it = categoryComboRows.begin(); it != categoryComboRows.end(); ++it ) { + if ( it.key() > row ) { + new_map.insert( it.key() - 1, it.data() ); + } + else + new_map.insert( it.key(), it.data() ); + } + categoryComboRows = new_map; +} + +void CategoryComboBox::modifyCategory( const Element &element ) +{ + for ( TQMap<int, int>::const_iterator it = categoryComboRows.begin(); it != categoryComboRows.end(); ++it ) { + if ( it.data() == element.id ) + changeItem( element.name, it.key() ); + } +} + +void CategoryComboBox::mergeCategories( int /*to_id*/, int from_id ) +{ + removeCategory( from_id ); +} + +int CategoryComboBox::findInsertionPoint( const TQString &name ) +{ + for ( int i = 1; i < count(); i++ ) { + if ( TQString::localeAwareCompare( name, text( i ) ) < 0 ) + return i; + } + + return count(); +} + +#include "categorycombobox.moc" diff --git a/src/widgets/categorycombobox.h b/src/widgets/categorycombobox.h new file mode 100644 index 0000000..4cc99aa --- /dev/null +++ b/src/widgets/categorycombobox.h @@ -0,0 +1,59 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef CATEGORYCOMBOBOX_H +#define CATEGORYCOMBOBOX_H + +#include <kcombobox.h> + +#include <tqmap.h> + +#include "datablocks/element.h" + +class RecipeDB; +class CategoryTree; + +class CategoryComboBox : public KComboBox +{ + TQ_OBJECT + +public: + CategoryComboBox( TQWidget *parent, RecipeDB *db ); + + void reload(); + int id( int row ); + +public slots: + void loadNextGroup(); + void loadPrevGroup(); + +protected: + virtual void popup(); + +private slots: + void createCategory( const Element &element, int /*parent_id*/ ); + void removeCategory( int id ); + void modifyCategory( const Element &element ); + void mergeCategories( int /*to_id*/, int from_id ); + + int findInsertionPoint( const TQString &name ); + +private: + void loadCategories( CategoryTree *categoryList, int &row ); + + RecipeDB *database; + TQMap<int, int> categoryComboRows; // Contains the category id for every given row in the category combobox + int m_offset; +}; + +#endif //CATEGORYCOMBOBOX_H + diff --git a/src/widgets/categorylistview.cpp b/src/widgets/categorylistview.cpp new file mode 100644 index 0000000..2ebe6e0 --- /dev/null +++ b/src/widgets/categorylistview.cpp @@ -0,0 +1,637 @@ + +/*************************************************************************** +* Copyright (C) 2004 by * +* Jason Kivlighn ([email protected]) * +* Unai Garro ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "categorylistview.h" + +#include <tdelocale.h> +#include <tdemessagebox.h> +#include <kiconloader.h> +#include <tdepopupmenu.h> +#include <tdeconfig.h> +#include <tdeglobal.h> +#include <kdebug.h> + +#include "backends/recipedb.h" +#include "datablocks/categorytree.h" +#include "dialogs/createcategorydialog.h" +#include "dialogs/dependanciesdialog.h" + +CategoryCheckListItem::CategoryCheckListItem( CategoryCheckListView* klv, const Element &category, bool _exclusive ) : TQCheckListItem( klv, TQString::null, TQCheckListItem::CheckBox ), CategoryItemInfo( category ), + locked( false ), + exclusive( _exclusive ), + m_listview(klv) +{ + setOn( false ); // Set unchecked by default +} + +CategoryCheckListItem::CategoryCheckListItem( TQListViewItem* it, const Element &category, bool _exclusive ) : TQCheckListItem( it, TQString::null, TQCheckListItem::CheckBox ), CategoryItemInfo( category ), + locked( false ), + exclusive( _exclusive ), + m_listview((CategoryCheckListView*)it->listView()) +{ + setOn( false ); // Set unchecked by default +} + +CategoryCheckListItem::CategoryCheckListItem( CategoryCheckListView* klv, TQListViewItem* it, const Element &category, bool _exclusive ) : TQCheckListItem( klv, it, TQString::null, TQCheckListItem::CheckBox ), CategoryItemInfo( category ), + locked( false ), + exclusive( _exclusive ), + m_listview(klv) +{ + setOn( false ); // Set unchecked by default +} + +TQString CategoryCheckListItem::text( int column ) const +{ + if ( column == 1 ) + return ( TQString::number( ctyStored.id ) ); + else + return ( ctyStored.name ); +} + +void CategoryCheckListItem::setText( int column, const TQString &text ) +{ + switch ( column ) { + case 0: + ctyStored.name = text; + break; + default: + break; + } +} + +void CategoryCheckListItem::stateChange( bool on ) +{ + m_listview->stateChange(this,on); + + if ( locked ) + return; + + if ( on && exclusive ) { + setParentsState( false ); + setChildrenState( false ); + } +} + +void CategoryCheckListItem::setChildrenState( bool on ) +{ + if ( !isPopulated() ) + return; + + for ( CategoryCheckListItem * cat_it = ( CategoryCheckListItem* ) firstChild(); cat_it; cat_it = ( CategoryCheckListItem* ) cat_it->nextSibling() ) { + cat_it->locked = true; + cat_it->setOn( on ); + cat_it->setChildrenState( on ); + cat_it->locked = false; + } +} + +void CategoryCheckListItem::setParentsState( bool on ) +{ + locked = true; + + CategoryCheckListItem *cat_it; + for ( cat_it = ( CategoryCheckListItem* ) parent(); cat_it; cat_it = ( CategoryCheckListItem* ) cat_it->parent() ) + cat_it->setOn( on ); + + locked = false; +} + + + + +CategoryListItem::CategoryListItem( TQListView* klv, const Element &category ) : TQListViewItem( klv ), + CategoryItemInfo(category) +{} + +CategoryListItem::CategoryListItem( TQListViewItem* it, const Element &category ) : TQListViewItem( it ), + CategoryItemInfo(category) +{} + +CategoryListItem::CategoryListItem( TQListView* klv, TQListViewItem* it, const Element &category ) : TQListViewItem( klv, it ), + CategoryItemInfo(category) +{} + +TQString CategoryListItem::text( int column ) const +{ + if ( column == 1 ) + return ( TQString::number( ctyStored.id ) ); + else + return ( ctyStored.name ); +} + +void CategoryListItem::setText( int column, const TQString &text ) +{ + if ( column == 0 ) + ctyStored.name = text; +} + + + +CategoryListView::CategoryListView( TQWidget *parent, RecipeDB *db ) : DBListViewBase( parent, db, db->categoryTopLevelCount() ), + m_item_to_delete(0) +{ + //connect( this, TQ_SIGNAL( spacePressed(TQListViewItem*) ), TQ_SLOT( open(TQListViewItem*) ) ); + //connect( this, TQ_SIGNAL( returnPressed(TQListViewItem*) ), TQ_SLOT( open(TQListViewItem*) ) ); + //connect( this, TQ_SIGNAL( executed(TQListViewItem*) ), TQ_SLOT( open(TQListViewItem*) ) ); + + connect( this, TQ_SIGNAL( expanded(TQListViewItem*) ), TQ_SLOT( open(TQListViewItem*) ) ); + + setRootIsDecorated( true ); + setAllColumnsShowFocus( true ); + setDefaultRenameAction( TQListView::Reject ); +} + +void CategoryListView::init() +{ + connect( database, TQ_SIGNAL( categoryCreated( const Element &, int ) ), TQ_SLOT( checkCreateCategory( const Element &, int ) ) ); + connect( database, TQ_SIGNAL( categoryRemoved( int ) ), TQ_SLOT( removeCategory( int ) ) ); + connect( database, TQ_SIGNAL( categoryModified( const Element & ) ), TQ_SLOT( modifyCategory( const Element & ) ) ); + connect( database, TQ_SIGNAL( categoryModified( int, int ) ), TQ_SLOT( modifyCategory( int, int ) ) ); + connect( database, TQ_SIGNAL( categoriesMerged( int, int ) ), TQ_SLOT( mergeCategories( int, int ) ) ); +} + +// (Re)loads the data from the database +void CategoryListView::load( int limit, int offset ) +{ + items_map.clear(); + + CategoryTree list; + CategoryTree *p_list = &list; + database->loadCachedCategories( &p_list, limit, offset, -1, false ); + + setTotalItems(p_list->count()); + + for ( CategoryTree * child_it = p_list->firstChild(); child_it; child_it = child_it->nextSibling() ) { + createCategory( child_it->category, -1 ); + } +} + +void CategoryListView::populate( TQListViewItem *item ) +{ + CategoryItemInfo *cat_item = dynamic_cast<CategoryItemInfo*>(item); + if ( !cat_item || cat_item->isPopulated() ) return; + + if ( item->firstChild() && item->firstChild()->rtti() != PSEUDOLISTITEM_RTTI ) + return; + + delete item->firstChild(); //delete the "pseudo item" + + int id = cat_item->categoryId(); + cat_item->setPopulated(true); + + CategoryTree categoryTree; + database->loadCategories( &categoryTree, -1, 0, id, false ); + + for ( CategoryTree * child_it = categoryTree.firstChild(); child_it; child_it = child_it->nextSibling() ) { + createCategory( child_it->category, id ); + } +} + +void CategoryListView::populateAll( TQListViewItem *parent ) +{ + if ( !parent ) + parent = firstChild(); + + for ( TQListViewItem *item = parent; item; item = item->nextSibling() ) { + populate( item ); + if ( item->firstChild() ) + populateAll( item->firstChild() ); + } +} + +void CategoryListView::open( TQListViewItem *item ) +{ + Q_ASSERT( item ); + if ( !item->firstChild() || item->firstChild()->rtti() != PSEUDOLISTITEM_RTTI ) return; + + populate(item); + + item->setOpen(true); +} + +void CategoryListView::checkCreateCategory( const Element &el, int parent_id ) +{ + if ( parent_id != -1 || handleElement(el.name) ) { //only create this category if the base class okays it; allow all non-top-level items + createCategory(el,parent_id); + } +} + +void CategoryListView::modifyCategory( const Element &category ) +{ + TQListViewItem * item = items_map[ category.id ]; + + if ( item ) + item->setText( 0, category.name ); +} + +void CategoryListView::modifyCategory( int id, int parent_id ) +{ + TQMap<int,TQListViewItem*>::iterator item_it = items_map.find(id); + if ( item_it != items_map.end() ) { + TQListViewItem *item = *item_it; + Q_ASSERT( item ); + + removeElement(item,false); + if ( !item->parent() ) + takeItem( item ); + else + item->parent() ->takeItem( item ); + + if ( parent_id == -1 ) { + insertItem(item); + createElement(item); + } + else { + TQMap<int,TQListViewItem*>::iterator parent_item_it = items_map.find(parent_id); + if ( parent_item_it != items_map.end() && + dynamic_cast<CategoryItemInfo*>(*parent_item_it)->isPopulated() ) { + (*parent_item_it)->insertItem( item ); + createElement(item); + } + else { + if ( !(*parent_item_it)->firstChild() ) + new PseudoListItem( *parent_item_it ); + + //removeElement() was already called on this item, so we just delete it + //we can't delete it just yet because this function is called by a slot + delete m_item_to_delete; + m_item_to_delete = item; + } + } + } +} + +void CategoryListView::mergeCategories( int id1, int id2 ) +{ + TQListViewItem * to_item = items_map[ id1 ]; + TQListViewItem *from_item = items_map[ id2 ]; + + CategoryItemInfo *info_item = dynamic_cast<CategoryItemInfo*>(to_item); + + if ( to_item && info_item->isPopulated() && from_item ) { + //note that this takes care of any recipes that may be children as well + TQListViewItem *next_sibling; + for ( TQListViewItem * it = from_item->firstChild(); it; it = next_sibling ) { + next_sibling = it->nextSibling(); //get the sibling before we move the item + + removeElement(it,false); + from_item->takeItem( it ); + + to_item->insertItem( it ); + createElement(it); + } + } + + removeCategory( id2 ); +} + + +StdCategoryListView::StdCategoryListView( TQWidget *parent, RecipeDB *db, bool editable ) : CategoryListView( parent, db ), + clipboard_item( 0 ), + clipboard_parent( 0 ) +{ + addColumn( i18n( "Category" ) ); + + TDEConfig *config = TDEGlobal::config(); + config->setGroup( "Advanced" ); + bool show_id = config->readBoolEntry( "ShowID", false ); + addColumn( i18n( "Id" ), show_id ? -1 : 0 ); + + if ( editable ) { + setRenameable( 0, true ); + setDragEnabled( true ); + setAcceptDrops( true ); + + TDEIconLoader *il = new TDEIconLoader; + + kpop = new TDEPopupMenu( this ); + kpop->insertItem( il->loadIcon( "document-new", TDEIcon::NoGroup, 16 ), i18n( "&Create" ), this, TQ_SLOT( createNew() ), CTRL + Key_C ); + kpop->insertItem( il->loadIcon( "edit-delete", TDEIcon::NoGroup, 16 ), i18n( "&Delete" ), this, TQ_SLOT( remove + () ), Key_Delete ); + kpop->insertItem( il->loadIcon( "edit", TDEIcon::NoGroup, 16 ), i18n( "&Rename" ), this, TQ_SLOT( rename() ), CTRL + Key_R ); + kpop->insertSeparator(); + kpop->insertItem( il->loadIcon( "edit-cut", TDEIcon::NoGroup, 16 ), i18n( "Cu&t" ), this, TQ_SLOT( cut() ), CTRL + Key_X ); + kpop->insertItem( il->loadIcon( "edit-paste", TDEIcon::NoGroup, 16 ), i18n( "&Paste" ), this, TQ_SLOT( paste() ), CTRL + Key_V ); + kpop->insertItem( il->loadIcon( "edit-paste", TDEIcon::NoGroup, 16 ), i18n( "Paste as Subcategory" ), this, TQ_SLOT( pasteAsSub() ), CTRL + SHIFT + Key_V ); + kpop->polish(); + + delete il; + + connect( kpop, TQ_SIGNAL( aboutToShow() ), TQ_SLOT( preparePopup() ) ); + connect( this, TQ_SIGNAL( contextMenu( TDEListView *, TQListViewItem *, const TQPoint & ) ), TQ_SLOT( showPopup( TDEListView *, TQListViewItem *, const TQPoint & ) ) ); + connect( this, TQ_SIGNAL( doubleClicked( TQListViewItem*, const TQPoint &, int ) ), TQ_SLOT( modCategory( TQListViewItem* ) ) ); + connect( this, TQ_SIGNAL( itemRenamed ( TQListViewItem* ) ), TQ_SLOT( saveCategory( TQListViewItem* ) ) ); + connect( this, TQ_SIGNAL( moved( TQListViewItem *, TQListViewItem *, TQListViewItem * ) ), TQ_SLOT( changeCategoryParent( TQListViewItem *, TQListViewItem *, TQListViewItem * ) ) ); + } +} + +StdCategoryListView::~StdCategoryListView() +{ + delete clipboard_item; +} + +void StdCategoryListView::setPixmap( const TQPixmap &icon ) +{ + m_folder_icon = icon; +} + +void StdCategoryListView::preparePopup() +{ + //only enable the paste items if clipboard_item isn't null + kpop->setItemEnabled( kpop->idAt( 5 ), clipboard_item ); + kpop->setItemEnabled( kpop->idAt( 6 ), clipboard_item ); +} + +void StdCategoryListView::showPopup( TDEListView * /*l*/, TQListViewItem *i, const TQPoint &p ) +{ + if ( i ) + kpop->exec( p ); +} + +void StdCategoryListView::createNew() +{ + ElementList categories; + database->loadCategories( &categories ); + CreateCategoryDialog* categoryDialog = new CreateCategoryDialog( this, categories ); + + if ( categoryDialog->exec() == TQDialog::Accepted ) { + TQString result = categoryDialog->newCategoryName(); + int subcategory = categoryDialog->subcategory(); + + //check bounds first + if ( checkBounds( result ) ) + database->createNewCategory( result, subcategory ); // Create the new category in the database + } + delete categoryDialog; +} + +void StdCategoryListView::remove + () +{ + TQListViewItem * item = currentItem(); + + if ( item ) { + int id = item->text( 1 ).toInt(); + + ElementList recipeDependancies; + database->findUseOfCategoryInRecipes( &recipeDependancies, id ); + + if ( recipeDependancies.isEmpty() ) { + switch ( KMessageBox::warningContinueCancel( this, i18n( "Are you sure you want to delete this category and all its subcategories?" ) ) ) { + case KMessageBox::Continue: + database->removeCategory( id ); + break; + } + return; + } + else { // need warning! + ListInfo info; + info.list = recipeDependancies; + info.name = i18n("Recipes"); + DependanciesDialog warnDialog( this, info, false ); + + if ( warnDialog.exec() == TQDialog::Accepted ) + database->removeCategory( id ); + } + } +} + +void StdCategoryListView::rename() +{ + TQListViewItem * item = currentItem(); + + if ( item ) + CategoryListView::rename( item, 0 ); +} + +void StdCategoryListView::cut() +{ + //restore a never used cut + if ( clipboard_item ) { + if ( clipboard_parent ) + clipboard_parent->insertItem( clipboard_item ); + else + insertItem( clipboard_item ); + clipboard_item = 0; + } + + TQListViewItem *item = currentItem(); + + if ( item ) { + clipboard_item = item; + clipboard_parent = item->parent(); + + if ( item->parent() ) + item->parent() ->takeItem( item ); + else + takeItem( item ); + } +} + +void StdCategoryListView::paste() +{ + TQListViewItem * item = currentItem(); + if ( item && clipboard_item ) { + if ( item->parent() ) + item->parent() ->insertItem( clipboard_item ); + else + insertItem( clipboard_item ); + + database->modCategory( clipboard_item->text( 1 ).toInt(), item->parent() ? item->parent() ->text( 1 ).toInt() : -1 ); + clipboard_item = 0; + } +} + +void StdCategoryListView::pasteAsSub() +{ + TQListViewItem * item = currentItem(); + + if ( item && clipboard_item ) { + item->insertItem( clipboard_item ); + database->modCategory( clipboard_item->text( 1 ).toInt(), item->text( 1 ).toInt() ); + clipboard_item = 0; + } +} + +void StdCategoryListView::changeCategoryParent( TQListViewItem *item, TQListViewItem * /*afterFirst*/, TQListViewItem * /*afterNow*/ ) +{ + int new_parent_id = -1; + if ( TQListViewItem * parent = item->parent() ) + new_parent_id = parent->text( 1 ).toInt(); + + int cat_id = item->text( 1 ).toInt(); + + disconnect( TQ_SIGNAL( moved( TQListViewItem *, TQListViewItem *, TQListViewItem * ) ) ); + database->modCategory( cat_id, new_parent_id ); + connect( this, TQ_SIGNAL( moved( TQListViewItem *, TQListViewItem *, TQListViewItem * ) ), TQ_SLOT( changeCategoryParent( TQListViewItem *, TQListViewItem *, TQListViewItem * ) ) ); +} + +void StdCategoryListView::removeCategory( int id ) +{ + TQListViewItem * item = items_map[ id ]; + + items_map.remove( id ); + removeElement(item); +} + +void StdCategoryListView::createCategory( const Element &category, int parent_id ) +{ + CategoryListItem * new_item = 0; + if ( parent_id == -1 ) { + new_item = new CategoryListItem( this, category ); + } + else { + CategoryListItem *parent = (CategoryListItem*)items_map[ parent_id ]; + + if ( parent ) { + if ( parent->isPopulated() ) + new_item = new CategoryListItem( parent, category ); + else if ( !parent->firstChild() ) { + new PseudoListItem( parent ); + parent->setOpen(true); + } + } + } + + if ( new_item ) { + items_map.insert( category.id, new_item ); + new_item->setPixmap( 0, m_folder_icon ); + createElement(new_item);//new TQListViewItem(new_item); + + CategoryTree list; + CategoryTree *p_list = &list; + database->loadCachedCategories( &p_list, 1, 0, category.id, false ); + + if ( p_list->firstChild() ) + new PseudoListItem( new_item ); + } +} + +void StdCategoryListView::modCategory( TQListViewItem* i ) +{ + if ( i ) + CategoryListView::rename( i, 0 ); +} + +void StdCategoryListView::saveCategory( TQListViewItem* i ) +{ + CategoryListItem * cat_it = ( CategoryListItem* ) i; + + if ( !checkBounds( cat_it->categoryName() ) ) { + reload(ForceReload); //reset the changed text + return ; + } + + int existing_id = database->findExistingCategoryByName( cat_it->categoryName() ); + int cat_id = cat_it->categoryId(); + if ( existing_id != -1 && existing_id != cat_id ) //category already exists with this label... merge the two + { + switch ( KMessageBox::warningContinueCancel( this, i18n( "This category already exists. Continuing will merge these two categories into one. Are you sure?" ) ) ) + { + case KMessageBox::Continue: { + database->mergeCategories( existing_id, cat_id ); + break; + } + default: + reload(ForceReload); + break; + } + } + else + database->modCategory( cat_id, cat_it->categoryName() ); +} + +bool StdCategoryListView::checkBounds( const TQString &name ) +{ + if ( name.length() > uint(database->maxCategoryNameLength()) ) { + KMessageBox::error( this, TQString( i18n( "Category name cannot be longer than %1 characters." ) ).arg( database->maxCategoryNameLength() ) ); + return false; + } + + return true; +} + + + +CategoryCheckListView::CategoryCheckListView( TQWidget *parent, RecipeDB *db, bool _exclusive, const ElementList &init_items_checked ) : CategoryListView( parent, db ), + exclusive(_exclusive) +{ + addColumn( i18n( "Category" ) ); + + TDEConfig *config = TDEGlobal::config(); + config->setGroup( "Advanced" ); + bool show_id = config->readBoolEntry( "ShowID", false ); + addColumn( i18n( "Id" ), show_id ? -1 : 0 ); + + for ( ElementList::const_iterator it = init_items_checked.begin(); it != init_items_checked.end(); ++it ) + m_selections.append(*it); +} + +void CategoryCheckListView::removeCategory( int id ) +{ + TQListViewItem * item = items_map[ id ]; + + items_map.remove( id ); + removeElement(item); +} + +void CategoryCheckListView::createCategory( const Element &category, int parent_id ) +{ + CategoryCheckListItem * new_item = 0; + if ( parent_id == -1 ) { + new_item = new CategoryCheckListItem( this, category, exclusive ); + } + else { + TQListViewItem *parent = items_map[ parent_id ]; + if ( parent ) + new_item = new CategoryCheckListItem( parent, category, exclusive ); + } + + if ( new_item ) { + items_map.insert( category.id, new_item ); + createElement(new_item); + + CategoryTree list; + CategoryTree *p_list = &list; + database->loadCachedCategories( &p_list, 1, 0, category.id, false ); + + if ( p_list->firstChild() ) + new PseudoListItem( new_item ); + + + new_item->setOpen( false ); + } +} + +void CategoryCheckListView::stateChange( CategoryCheckListItem* it, bool on ) +{ + if ( !reloading() ) { + if ( on ) + m_selections.append(it->element()); + else + m_selections.remove(it->element()); + } +} + +void CategoryCheckListView::load( int limit, int offset ) +{ + CategoryListView::load(limit,offset); + + for ( TQValueList<Element>::const_iterator it = m_selections.begin(); it != m_selections.end(); ++it ) { + TQCheckListItem * item = ( TQCheckListItem* ) findItem( TQString::number( (*it).id ), 1 ); + if ( item ) { + item->setOn(true); + } + } +} + +#include "categorylistview.moc" diff --git a/src/widgets/categorylistview.h b/src/widgets/categorylistview.h new file mode 100644 index 0000000..1dcabba --- /dev/null +++ b/src/widgets/categorylistview.h @@ -0,0 +1,288 @@ +/*************************************************************************** +* Copyright (C) 2004 by * +* Jason Kivlighn ([email protected]) * +* Unai Garro ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef CATEGORYLISTVIEW_H +#define CATEGORYLISTVIEW_H + +#include <tqmap.h> +#include <tqpixmap.h> + +#include "dblistviewbase.h" + +#include "datablocks/elementlist.h" + +class TDEPopupMenu; + +class RecipeDB; +class CategoryTree; +class CategoryCheckListView; + +#define CATEGORYCHECKLISTITEM_RTTI 1005 +#define CATEGORYLISTITEM_RTTI 1001 +#define PSEUDOLISTITEM_RTTI 1008 + +/** Category listitems inherit this class to provide a common interface for accessing this information. + */ +class CategoryItemInfo +{ +public: + CategoryItemInfo( const Element &category ) : ctyStored( category ), populated(false){} + bool isPopulated() const { return populated; } + void setPopulated( bool b ){ populated = b; } + + Element element() const + { + return ctyStored; + } + + int categoryId() const + { + return ctyStored.id; + } + TQString categoryName() const + { + return ctyStored.name; + } + +protected: + Element ctyStored; + +private: + bool populated; +}; + +class CategoryCheckListItem : public TQCheckListItem, public CategoryItemInfo +{ +public: + CategoryCheckListItem( CategoryCheckListView* klv, const Element &category, bool exclusive = true ); + CategoryCheckListItem( TQListViewItem* it, const Element &category, bool exclusive = true ); + CategoryCheckListItem( CategoryCheckListView* klv, TQListViewItem* it, const Element &category, bool exclusive = true ); + + virtual TQString text( int column ) const; + virtual void setText( int column, const TQString &text ); + + int rtti() const + { + return CATEGORYCHECKLISTITEM_RTTI; + } + +protected: + virtual void stateChange( bool ); + void setChildrenState( bool ); + void setParentsState( bool ); + + bool locked; + bool exclusive; + +private: + CategoryCheckListView *m_listview; +}; + + +class CategoryListItem : public TQListViewItem, public CategoryItemInfo +{ +public: + CategoryListItem( TQListView* klv, const Element &category ); + CategoryListItem( TQListViewItem* it, const Element &category ); + CategoryListItem( TQListView* klv, TQListViewItem* it, const Element &category ); + + virtual TQString text( int column ) const; + virtual void setText( int column, const TQString &text ); + + int rtti() const + { + return CATEGORYLISTITEM_RTTI; + } +}; + + + +class CategoryListView : public DBListViewBase +{ + TQ_OBJECT + +public: + CategoryListView( TQWidget *parent, RecipeDB * ); + + void populateAll( TQListViewItem *parent = 0 ); + +public slots: + void open( TQListViewItem *item ); + +protected: + virtual void init(); + + virtual void load( int limit, int offset ); + + /** so that it allows dropping into + * subchildren that aren't expandable. The code is taken from TDE's TDEListView with + * one line commented out. + */ + void findDrop( const TQPoint &pos, TQListViewItem *&parent, TQListViewItem *&after ) + { + TQPoint p ( contentsToViewport( pos ) ); + + // Get the position to put it in + TQListViewItem *atpos = itemAt( p ); + + TQListViewItem *above; + if ( !atpos ) // put it at the end + above = lastItem(); + else { + // Get the closest item before us ('atpos' or the one above, if any) + if ( p.y() - itemRect( atpos ).topLeft().y() < ( atpos->height() / 2 ) ) + above = atpos->itemAbove(); + else + above = atpos; + } + + if ( above ) { + // if above has children, I might need to drop it as the first item there + + if ( above->firstChild() && above->isOpen() ) { + parent = above; + after = 0; + return ; + } + + // Now, we know we want to go after "above". But as a child or as a sibling ? + // We have to ask the "above" item if it accepts children. + // ### NOTE: Here is the one line commented out so that "above" always accepts children + //if (above->isExpandable()) + { + // The mouse is sufficiently on the right ? - doesn't matter if 'above' has visible children + if ( p.x() >= depthToPixels( above->depth() + 1 ) || + ( above->isOpen() && above->childCount() > 0 ) ) + { + parent = above; + after = 0L; + return ; + } + } + + // Ok, there's one more level of complexity. We may want to become a new + // sibling, but of an upper-level group, rather than the "above" item + TQListViewItem * betterAbove = above->parent(); + TQListViewItem * last = above; + while ( betterAbove ) { + // We are allowed to become a sibling of "betterAbove" only if we are + // after its last child + if ( last->nextSibling() == 0 ) { + if ( p.x() < depthToPixels ( betterAbove->depth() + 1 ) ) + above = betterAbove; // store this one, but don't stop yet, there may be a better one + else + break; // not enough on the left, so stop + last = betterAbove; + betterAbove = betterAbove->parent(); // up one level + } + else + break; // we're among the child of betterAbove, not after the last one + } + } + // set as sibling + after = above; + parent = after ? after->parent() : 0L ; + } + +protected slots: + virtual void removeCategory( int id ) = 0; + virtual void createCategory( const Element &category, int parent_id ) = 0; + virtual void modifyCategory( const Element &category ); + virtual void modifyCategory( int id, int parent_id ); + virtual void mergeCategories( int id1, int id2 ); + + virtual void checkCreateCategory( const Element &, int parent_id ); + virtual void populate( TQListViewItem *item ); + + TQMap<int, TQListViewItem*> items_map; + +private: + TQListViewItem *m_item_to_delete; +}; + + +class StdCategoryListView : public CategoryListView +{ + TQ_OBJECT + +public: + StdCategoryListView( TQWidget *parent, RecipeDB *, bool editable = false ); + ~StdCategoryListView(); + +protected: + virtual void removeCategory( int id ); + virtual void createCategory( const Element &category, int parent_id ); + + void setPixmap( const TQPixmap &pixmap ); + +private slots: + void preparePopup(); + void showPopup( TDEListView *, TQListViewItem *, const TQPoint & ); + + void createNew(); + void remove + (); + void rename(); + void cut(); + void paste(); + void pasteAsSub(); + + void changeCategoryParent( TQListViewItem *item, TQListViewItem * /*afterFirst*/, TQListViewItem * /*afterNow*/ ); + + void modCategory( TQListViewItem* i ); + void saveCategory( TQListViewItem* i ); + +private: + bool checkBounds( const TQString &name ); + + TDEPopupMenu *kpop; + TQListViewItem *clipboard_item; + TQListViewItem *clipboard_parent; + + TQPixmap m_folder_icon; +}; + + +class CategoryCheckListView : public CategoryListView +{ + TQ_OBJECT + +public: + CategoryCheckListView( TQWidget *parent, RecipeDB *, bool exclusive=true, const ElementList &init_items_checked = ElementList() ); + + virtual void stateChange( CategoryCheckListItem*, bool ); + + ElementList selections() const{ return m_selections; } + +protected: + virtual void removeCategory( int id ); + virtual void createCategory( const Element &category, int parent_id ); + + virtual void load( int limit, int offset ); + + bool exclusive; + +private: + ElementList m_selections; +}; + + +class PseudoListItem : public TQListViewItem +{ +public: + PseudoListItem( TQListView* lv ) : TQListViewItem(lv){} + PseudoListItem( TQListViewItem* it ) : TQListViewItem(it){} + +protected: + int rtti() const { return PSEUDOLISTITEM_RTTI; } +}; + +#endif //CATEGORYLISTVIEW_H diff --git a/src/widgets/conversiontable.cpp b/src/widgets/conversiontable.cpp new file mode 100644 index 0000000..a48b67e --- /dev/null +++ b/src/widgets/conversiontable.cpp @@ -0,0 +1,426 @@ +/*************************************************************************** +* Copyright (C) 2003-2004 by * +* Unai Garro ([email protected]) * +* * +* Copyright (C) 2003-2006 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "conversiontable.h" +#include "datablocks/mixednumber.h" +#include "widgets/fractioninput.h" + +#include <tqtooltip.h> + +#include <tdeglobal.h> +#include <tdelocale.h> + +class ConversionTableToolTip : public TQToolTip +{ +public: + ConversionTableToolTip( ConversionTable *t ) : TQToolTip( t->viewport() ), + table( t ) + {} + + void maybeTip( const TQPoint &pos ) + { + if ( !table ) + return ; + + TQPoint cp = table->viewportToContents( pos ); + + int row = table->rowAt( cp.y() ); + int col = table->columnAt( cp.x() ); + + if ( row == col ) + return ; + + TQString row_unit = table->verticalHeader() ->label( row ); + TQString col_unit = table->horizontalHeader() ->label( col ); + TQString text = table->text( row, col ); + if ( text.isEmpty() ) + text = "X"; //### Is this i18n friendly??? + + TQRect cr = table->cellGeometry( row, col ); + cr.moveTopLeft( table->contentsToViewport( cr.topLeft() ) ); + tip( cr, TQString( "1 %1 = %2 %3" ).arg( row_unit ).arg( text ).arg( col_unit ) ); + } + +private: + ConversionTable *table; +}; + +ConversionTable::ConversionTable( TQWidget* parent, int maxrows, int maxcols ) : TQTable( maxrows, maxcols, parent, "table" ) +{ + editBoxValue = -1; + items.setAutoDelete( true ); + widgets.setAutoDelete( true ); + + ( void ) new ConversionTableToolTip( this ); +} + +ConversionTable::~ConversionTable() +{} +#include <kdebug.h> +void ConversionTable::unitRemoved( int id ) +{ + int index = *unitIDs.find( id ); + kdDebug() << "index:" << index << endl; + removeRow( index ); + removeColumn( index ); + kdDebug() << "done" << endl; +} + +void ConversionTable::unitCreated( const Unit &unit ) +{ + insertColumns( numCols() ); + insertRows( numRows() ); + unitIDs.append( unit.id ); + horizontalHeader() ->setLabel( numRows() - 1, unit.name ); + verticalHeader() ->setLabel( numCols() - 1, unit.name ); +} + +TQTableItem* ConversionTable::item( int r, int c ) const +{ + return items.find( indexOf( r, c ) ); +} + +void ConversionTable::setItem( int r, int c, TQTableItem *i ) +{ + items.replace( indexOf( r, c ), i ); + i->setRow( r ); // Otherwise the item + i->setCol( c ); //doesn't know where it is! + updateCell( r, c ); +} + +void ConversionTable::clearCell( int r, int c ) +{ + items.remove( indexOf( r, c ) ); +} + +void ConversionTable::takeItem( TQTableItem *item ) +{ + items.setAutoDelete( false ); + items.remove( indexOf( item->row(), item->col() ) ); + items.setAutoDelete( true ); +} + +void ConversionTable::insertWidget( int r, int c, TQWidget *w ) +{ + widgets.replace( indexOf( r, c ), w ); +} + +TQWidget* ConversionTable::cellWidget( int r, int c ) const +{ + return widgets.find( indexOf( r, c ) ); +} + +void ConversionTable::clearCellWidget( int r, int c ) +{ + TQWidget * w = widgets.take( indexOf( r, c ) ); + if ( w ) + w->deleteLater(); +} + + +ConversionTableItem::ConversionTableItem( TQTable *t, EditType et ) : TQTableItem( t, et, TQString::null ) +{ + //�we�do�not�want�this�item�to�be�replaced + setReplaceable( false ); +} + +void ConversionTableItem::paint( TQPainter *p, const TQColorGroup &cg, const TQRect &cr, bool selected ) +{ + TQColorGroup g( cg ); + + // Draw in gray all those cells which are not editable + + if ( row() == col() ) + g.setColor( TQColorGroup::Base, gray ); + TQTableItem::paint( p, g, cr, selected ); +} + +TQWidget* ConversionTableItem::createEditor() const +{ + FractionInput *editor = new FractionInput( table()->viewport(), MixedNumber::DecimalFormat ); + + MixedNumber current = MixedNumber::fromString(text()); + if ( current.toDouble() > 1e-8 ) + editor->setValue( current, 0 ); + + return editor; +} + +void ConversionTableItem::setContentFromEditor( TQWidget *w ) +{ + // the�user changed the value of the combobox, so synchronize the + // value of the item (its text), with the value of the combobox + if ( w->inherits( "FractionInput" ) ) { + FractionInput* editor = ( FractionInput* ) w; + if ( editor->isInputValid() && !editor->isEmpty() && editor->value().toDouble() > 1e-6 ) { + setText( editor->value().toString(MixedNumber::DecimalFormat) ); + emit ratioChanged( row(), col(), editor->value().toDouble() ); // Signal to store + } + else { + setText( TQString::null ); + emit ratioRemoved( row(), col() ); + } + } + else + TQTableItem::setContentFromEditor( w ); +} + +void ConversionTableItem::setText( const TQString &s ) +{ + TQTableItem::setText( s ); +} +TQString ConversionTable::text( int r, int c ) const // without this function, the usual (text(r,c)) won't work +{ + if ( item( r, c ) ) + return item( r, c ) ->text(); //Note that item(r,c) was reimplemented here for large sparse tables... + else + return TQString::null; +} + +void ConversionTable::initTable() +{ + + for ( int r = 0;r < numRows();r++ ) { + this->createNewItem( r, r, 1.0 ); + item( r, r ) ->setEnabled( false ); // Diagonal is not editable + } +} + +void ConversionTable::createNewItem( int r, int c, double amount ) +{ + + ConversionTableItem * ci = new ConversionTableItem( this, TQTableItem::WhenCurrent ); + ci->setText( beautify( TDEGlobal::locale() ->formatNumber( amount, 5 ) ) ); + setItem( r, c, ci ); + // connect signal (forward) to know when it's actually changed + connect( ci, TQ_SIGNAL( ratioChanged( int, int, double ) ), this, TQ_SIGNAL( ratioChanged( int, int, double ) ) ); + connect( ci, TQ_SIGNAL( ratioRemoved( int, int ) ), this, TQ_SIGNAL( ratioRemoved( int, int ) ) ); + connect( ci, TQ_SIGNAL( signalRepaintCell( int, int ) ), this, TQ_SLOT( repaintCell( int, int ) ) ); +} + +void ConversionTable::setUnitIDs( const IDList &idList ) +{ + unitIDs = idList; +} + +void ConversionTable::setRatio( int ingID1, int ingID2, double ratio ) +{ + int indexID1 = unitIDs.findIndex( ingID1 ); + int indexID2 = unitIDs.findIndex( ingID2 ); + + createNewItem( indexID1, indexID2, ratio ); +} + + +int ConversionTable::getUnitID( int rc ) +{ + return ( *( unitIDs.at( rc ) ) ); +} + +TQWidget * ConversionTable::beginEdit ( int row, int col, bool replace ) +{ + // If there's no item, create it first. + if ( !item( row, col ) ) { + createNewItem( row, col, 0 ); + } + + // Then call normal beginEdit + return TQTable::beginEdit( row, col, replace ); +} + +void ConversionTableItem::setTextAndSave( const TQString &s ) +{ + setText( s ); // Change text + emit signalRepaintCell( row(), col() ); // Indicate to update the cell to the table. Otherwise it's not repainted + emit ratioChanged( row(), col(), s.toDouble() ); // Signal to store +} + +void ConversionTable::repaintCell( int r, int c ) +{ + TQTable::updateCell( r, c ); +} + +void ConversionTable::resize( int r, int c ) +{ + setNumRows( r ); + setNumCols( c ); + initTable(); +} + +void ConversionTable::clear( void ) +{ + items.clear(); + widgets.clear(); + unitIDs.clear(); + resize( 0, 0 ); + +} + +//TODO this is incomplete/wrong +void ConversionTable::swapRows( int row1, int row2, bool /*swapHeader*/ ) +{ + //if ( swapHeader ) + //((TQTableHeader*)verticalHeader())->swapSections( row1, row2, FALSE ); + + TQPtrVector<TQTableItem> tmpContents; + tmpContents.resize( numCols() ); + TQPtrVector<TQWidget> tmpWidgets; + tmpWidgets.resize( numCols() ); + int i; + + items.setAutoDelete( FALSE ); + widgets.setAutoDelete( FALSE ); + for ( i = 0; i < numCols(); ++i ) { + TQTableItem *i1, *i2; + i1 = item( row1, i ); + i2 = item( row2, i ); + if ( i1 || i2 ) { + tmpContents.insert( i, i1 ); + items.remove( indexOf( row1, i ) ); + items.insert( indexOf( row1, i ), i2 ); + items.remove( indexOf( row2, i ) ); + items.insert( indexOf( row2, i ), tmpContents[ i ] ); + if ( items[ indexOf( row1, i ) ] ) + items[ indexOf( row1, i ) ] ->setRow( row1 ); + if ( items[ indexOf( row2, i ) ] ) + items[ indexOf( row2, i ) ] ->setRow( row2 ); + } + + TQWidget *w1, *w2; + w1 = cellWidget( row1, i ); + w2 = cellWidget( row2, i ); + if ( w1 || w2 ) { + tmpWidgets.insert( i, w1 ); + widgets.remove( indexOf( row1, i ) ); + widgets.insert( indexOf( row1, i ), w2 ); + widgets.remove( indexOf( row2, i ) ); + widgets.insert( indexOf( row2, i ), tmpWidgets[ i ] ); + } + } + items.setAutoDelete( FALSE ); + widgets.setAutoDelete( TRUE ); + + //updateRowWidgets( row1 ); + //updateRowWidgets( row2 ); + /* + if ( curRow == row1 ) + curRow = row2; + else if ( curRow == row2 ) + curRow = row1; + if ( editRow == row1 ) + editRow = row2; + else if ( editRow == row2 ) + editRow = row1;*/ +} + +//TODO this is incomplete/wrong +void ConversionTable::swapColumns( int col1, int col2, bool /*swapHeader*/ ) +{ + //if ( swapHeader ) + //((TQTableHeader*)horizontalHeader())->swapSections( col1, col2, FALSE ); + + TQPtrVector<TQTableItem> tmpContents; + tmpContents.resize( numRows() ); + TQPtrVector<TQWidget> tmpWidgets; + tmpWidgets.resize( numRows() ); + int i; + + items.setAutoDelete( FALSE ); + widgets.setAutoDelete( FALSE ); + for ( i = 0; i < numRows(); ++i ) { + TQTableItem *i1, *i2; + i1 = item( i, col1 ); + i2 = item( i, col2 ); + if ( i1 || i2 ) { + tmpContents.insert( i, i1 ); + items.remove( indexOf( i, col1 ) ); + items.insert( indexOf( i, col1 ), i2 ); + items.remove( indexOf( i, col2 ) ); + items.insert( indexOf( i, col2 ), tmpContents[ i ] ); + if ( items[ indexOf( i, col1 ) ] ) + items[ indexOf( i, col1 ) ] ->setCol( col1 ); + if ( items[ indexOf( i, col2 ) ] ) + items[ indexOf( i, col2 ) ] ->setCol( col2 ); + } + + TQWidget *w1, *w2; + w1 = cellWidget( i, col1 ); + w2 = cellWidget( i, col2 ); + if ( w1 || w2 ) { + tmpWidgets.insert( i, w1 ); + widgets.remove( indexOf( i, col1 ) ); + widgets.insert( indexOf( i, col1 ), w2 ); + widgets.remove( indexOf( i, col2 ) ); + widgets.insert( indexOf( i, col2 ), tmpWidgets[ i ] ); + } + } + items.setAutoDelete( FALSE ); + widgets.setAutoDelete( TRUE ); + + columnWidthChanged( col1 ); + columnWidthChanged( col2 ); + /* + if ( curCol == col1 ) + curCol = col2; + else if ( curCol == col2 ) + curCol = col1; + if ( editCol == col1 ) + editCol = col2; + else if ( editCol == col2 ) + editCol = col1;*/ +} + +//TODO this is incomplete/wrong +void ConversionTable::swapCells( int row1, int col1, int row2, int col2 ) +{ + items.setAutoDelete( FALSE ); + widgets.setAutoDelete( FALSE ); + TQTableItem *i1, *i2; + i1 = item( row1, col1 ); + i2 = item( row2, col2 ); + if ( i1 || i2 ) { + TQTableItem * tmp = i1; + items.remove( indexOf( row1, col1 ) ); + items.insert( indexOf( row1, col1 ), i2 ); + items.remove( indexOf( row2, col2 ) ); + items.insert( indexOf( row2, col2 ), tmp ); + if ( items[ indexOf( row1, col1 ) ] ) { + items[ indexOf( row1, col1 ) ] ->setRow( row1 ); + items[ indexOf( row1, col1 ) ] ->setCol( col1 ); + } + if ( items[ indexOf( row2, col2 ) ] ) { + items[ indexOf( row2, col2 ) ] ->setRow( row2 ); + items[ indexOf( row2, col2 ) ] ->setCol( col2 ); + } + } + + TQWidget *w1, *w2; + w1 = cellWidget( row1, col1 ); + w2 = cellWidget( row2, col2 ); + if ( w1 || w2 ) { + TQWidget * tmp = w1; + widgets.remove( indexOf( row1, col1 ) ); + widgets.insert( indexOf( row1, col1 ), w2 ); + widgets.remove( indexOf( row2, col2 ) ); + widgets.insert( indexOf( row2, col2 ), tmp ); + } + + //updateRowWidgets( row1 ); + //updateRowWidgets( row2 ); + //updateColWidgets( col1 ); + //updateColWidgets( col2 ); + items.setAutoDelete( FALSE ); + widgets.setAutoDelete( TRUE ); +} + +#include "conversiontable.moc" diff --git a/src/widgets/conversiontable.h b/src/widgets/conversiontable.h new file mode 100644 index 0000000..16f761c --- /dev/null +++ b/src/widgets/conversiontable.h @@ -0,0 +1,101 @@ +/*************************************************************************** +* Copyright (C) 2003-2004 by * +* Unai Garro ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ +#ifndef CONVERSIONTABLE_H +#define CONVERSIONTABLE_H +#include <tqstring.h> +#include <tqtable.h> +#include <tqobject.h> + +#include "datablocks/unitratio.h" +#include "datablocks/elementlist.h" +#include "datablocks/unit.h" + +/** +@author Unai Garro +*/ + + +class ConversionTable: public TQTable +{ + TQ_OBJECT +public: + + ConversionTable( TQWidget* parent, int maxrows, int maxcols ); + ~ConversionTable(); + void createNewItem( int r, int c, double amount ); + void setUnitIDs( const IDList &idList ); + void setRatio( int ingID1, int ingID2, double ratio ); + void setRatio( const UnitRatio &r ) + { + setRatio( r.uID1, r.uID2, r.ratio ); + } + int getUnitID( int rc ); + TQString text( int r, int c ) const; //Reimplement, otherwise it won't work this way + void resize( int r, int c ); + void clear( void ); +private: + + //Internal Variables + double editBoxValue; + TQIntDict<TQTableItem> items; + TQIntDict<TQWidget> widgets; + IDList unitIDs; // unit ID list to know the units by ID, not name + //Internal Methods + void resizeData( int ) + {} + ; + TQTableItem *item( int r, int c ) const; + void setItem( int r, int c, TQTableItem *i ); + void clearCell( int r, int c ); + void takeItem( TQTableItem *item ); + void insertWidget( int r, int c, TQWidget *w ); + TQWidget *cellWidget( int r, int c ) const; + void clearCellWidget( int r, int c ); + void initTable(); + void swapRows( int, int, bool ); + void swapColumns( int, int, bool ); + void swapCells( int, int, int, int ); +protected: + TQWidget* beginEdit ( int row, int col, bool replace ); + +private slots: + void repaintCell( int r, int c ); + + void unitRemoved( int ); + void unitCreated( const Unit& ); +signals: + void ratioChanged( int row, int col, double value ); + void ratioRemoved( int row, int col ); +}; + +class ConversionTableItem: public TQObject, public TQTableItem +{ + TQ_OBJECT +public: + ConversionTableItem( TQTable *t, EditType et ); + TQWidget *createEditor() const; + void setContentFromEditor( TQWidget *w ); + void setText( const TQString &s ); + void paint( TQPainter *p, const TQColorGroup &cg, const TQRect &cr, bool selected ); + void setTextAndSave( const TQString &s ); + int alignment() const + { + return TQt::AlignRight; + } +signals: + void ratioChanged( int row, int col, double value ); + void ratioRemoved( int row, int col ); + void signalRepaintCell( int r, int c ); + +}; + + +#endif diff --git a/src/widgets/criteriacombobox.cpp b/src/widgets/criteriacombobox.cpp new file mode 100644 index 0000000..8fd911a --- /dev/null +++ b/src/widgets/criteriacombobox.cpp @@ -0,0 +1,49 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "criteriacombobox.h" + +#include <tqlistbox.h> + +#include "backends/recipedb.h" +#include "datablocks/elementlist.h" + +CriteriaComboBox::CriteriaComboBox( bool b, TQWidget *parent, RecipeDB *db ) : KComboBox( b, parent ), + database( db ) +{ + connect( db, TQ_SIGNAL(ratingCriteriaCreated(const Element &)), this, TQ_SLOT(addCriteria(const Element &)) ); +} + +void CriteriaComboBox::addCriteria( const Element &criteria ) +{ + idMap.insert(count(),criteria.id); + + insertItem(criteria.name); + completionObject()->addItem(criteria.name); +} + +void CriteriaComboBox::reload() +{ + ElementList criteriaList; + database->loadRatingCriterion( &criteriaList ); + + clear(); + + for ( ElementList::const_iterator it = criteriaList.begin(); it != criteriaList.end(); ++it ) { + addCriteria((*it)); + } +} + +int CriteriaComboBox::criteriaID( int index ) +{ + return idMap[index]; +} + +#include "criteriacombobox.moc" diff --git a/src/widgets/criteriacombobox.h b/src/widgets/criteriacombobox.h new file mode 100644 index 0000000..4e35bb4 --- /dev/null +++ b/src/widgets/criteriacombobox.h @@ -0,0 +1,41 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef CRITERIACOMBOBOX_H +#define CRITERIACOMBOBOX_H + +#include <tqmap.h> + +#include <kcombobox.h> + +#include "datablocks/element.h" + +class RecipeDB; + +class CriteriaComboBox : public KComboBox +{ + TQ_OBJECT + +public: + CriteriaComboBox( bool, TQWidget *parent, RecipeDB *db ); + + void reload(); + int criteriaID( int index ); + +protected slots: + void addCriteria( const Element &criteria ); + +private: + RecipeDB *database; + TQMap< int, int > idMap; +}; + +#endif //CRITERIACOMBOBOX_H + diff --git a/src/widgets/dblistviewbase.cpp b/src/widgets/dblistviewbase.cpp new file mode 100644 index 0000000..4a22330 --- /dev/null +++ b/src/widgets/dblistviewbase.cpp @@ -0,0 +1,334 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "dblistviewbase.h" + +#include <tdeapplication.h> +#include <tdeconfig.h> +#include <kcursor.h> +#include <kdebug.h> +#include <tdeglobal.h> +#include <tdelocale.h> +#include <kprogress.h> + + +//These two classes are used to identify the "Next" and "Prev" items, which are identified through rtti(). This also prevents renaming, even if it is enabled. +class PrevListViewItem : public TQListViewItem +{ +public: + PrevListViewItem( TQListView *parent ) : TQListViewItem(parent){} + + virtual int rtti() const { return PREVLISTITEM_RTTI; } + + TQString text( int c ) const { + if ( c == 0 ) { + return TQString("<< %1").arg(i18n("Previous")); + } + else + return TQString::null; + } +}; + +class NextListViewItem : public TQListViewItem +{ +public: + NextListViewItem( TQListView *parent, TQListViewItem *after ) : TQListViewItem(parent,after){} + + virtual int rtti() const { return NEXTLISTITEM_RTTI; } + + TQString text( int c ) const { + if ( c == 0 ) { + return TQString("%1 >>").arg(i18n("Next")); + } + else + return TQString::null; + } +}; + +DBListViewBase::DBListViewBase( TQWidget *parent, RecipeDB *db, int t ) : TDEListView(parent), + database(db), + curr_limit(-1), + curr_offset(0), + total(t), + bulk_load(false), + delete_me_later(0), + m_progress(0), + m_totalSteps(0) +{ + setSorting(-1); + + if ( curr_limit == -1 ) { //only use the default limit if a subclass hasn't given curr_limit its own value + TDEConfig * config = TDEGlobal::config();config->setGroup( "Performance" ); + curr_limit = config->readNumEntry( "Limit", -1 ); + } + + connect(this,TQ_SIGNAL(executed(TQListViewItem*)),TQ_SLOT(slotDoubleClicked(TQListViewItem*))); +} + +DBListViewBase::~DBListViewBase() +{ + delete m_progress; +} + +void DBListViewBase::activatePrev() +{ + if ( curr_offset != 0 ) { + curr_offset -= curr_limit; + if ( curr_offset < 0 ) + curr_offset = 0; + + reload(ForceReload); + emit prevGroupLoaded(); + } +} + +void DBListViewBase::activateNext() +{ + curr_offset += curr_limit; + + reload(ForceReload); + emit nextGroupLoaded(); +} + +void DBListViewBase::rename( TQListViewItem *it, int c ) +{ + if ( it->rtti() == PREVLISTITEM_RTTI || it->rtti() == NEXTLISTITEM_RTTI ) { + return; + } + + TDEListView::rename(it,c); +} + +void DBListViewBase::slotDoubleClicked( TQListViewItem *it ) +{ + //we can't delete the item the was double clicked + //and yet these functions will clear() the listview. + //We'll take the item from the view so it isn't deleted + //and delete it ourselves. + delete delete_me_later; delete_me_later = 0; + + if ( it->rtti() == PREVLISTITEM_RTTI ) { + delete_me_later = it; + takeItem(it); + activatePrev(); + } + else if ( it->rtti() == NEXTLISTITEM_RTTI ) { + delete_me_later = it; + takeItem(it); + activateNext(); + } +} + +void DBListViewBase::keyPressEvent( TQKeyEvent *k ) +{ + if ( k->state() == TQt::ShiftButton ) { + switch ( k->key() ) { + case TQt::Key_N: { + if ( curr_offset + curr_limit >= total || curr_limit == -1 ) { + k->accept(); + return; + } + + kapp->processEvents(); //if auto-repeating, user won't otherwise see change in the listview + activateNext(); + k->accept(); + break; + } + case TQt::Key_P: { + kapp->processEvents(); //if auto-repeating, user won't otherwise see change in the listview + activatePrev(); + k->accept(); + break; + } + default: break; + } + } + + TDEListView::keyPressEvent(k); +} + +void DBListViewBase::reload( ReloadFlags flag ) +{ + if ( flag == ForceReload || (!firstChild() && flag == Load) || (firstChild() && flag == ReloadIfPopulated) ) { + TDEApplication::setOverrideCursor( KCursor::waitCursor() ); + + init(); + + //m_progress = new KProgressDialog(this,0,TQString::null,i18n("Loading..."),true); + //m_progress->setAllowCancel(false); + //m_progress->progressBar()->setPercentageVisible(false); + //m_progress->progressBar()->setTotalSteps( m_totalSteps ); + //m_progress->show(); + //kapp->processEvents(); + + //reset some things + clear(); + lastElementMap.clear(); + + bulk_load=true; + load(curr_limit,curr_offset); + bulk_load=false; + + if ( curr_limit != -1 && curr_offset + curr_limit < total ) + new NextListViewItem(this,lastElementMap[0]); + + if ( curr_offset != 0 ) + new PrevListViewItem(this); + + //delete m_progress; m_progress = 0; + + TDEApplication::restoreOverrideCursor(); + } +} + +void DBListViewBase::setTotalItems(int i) +{ + m_totalSteps = i; + if ( m_progress ) { + m_progress->progressBar()->setTotalSteps( m_totalSteps ); + } +} + +void DBListViewBase::createElement( TQListViewItem *it ) +{ + Q_ASSERT(it); + + TQListViewItem *lastElement; + TQMap<TQListViewItem*,TQListViewItem*>::iterator map_it = lastElementMap.find(it->parent()); + if ( map_it != lastElementMap.end() ) { + lastElement = map_it.data(); + } + else + lastElement = 0; + + if ( bulk_load ) { //this can be much faster if we know the elements are already in order + if ( lastElement ) it->moveItem(lastElement); + lastElementMap.insert(it->parent(),it); + if ( m_progress ) { m_progress->progressBar()->advance(1); } + } + else { + if ( lastElement == 0 ) { + lastElementMap.insert(it->parent(),it); + } + else { + + int c = 0;//FIXME: the column used should be variable (set by a subclass) + + if ( it->parent() == 0 ) { + //start it out below the "Prev" item... currently it will be at firstChild() + if ( firstChild()->nextSibling() && + ( firstChild()->nextSibling()->rtti() == PREVLISTITEM_RTTI || + firstChild()->nextSibling()->rtti() == 1006 ) ) { //A hack to skip the Uncategorized item + it->moveItem( firstChild()->nextSibling() ); + } + } + + if ( TQString::localeAwareCompare(it->text(c),lastElement->text(c)) >= 0 ) { + it->moveItem(lastElement); + lastElementMap.insert(it->parent(),it); + } + else { + TQListViewItem *last_it = 0; + + for ( TQListViewItem *search_it = it; search_it; search_it = search_it->nextSibling() ) { + if ( search_it->rtti() == NEXTLISTITEM_RTTI ) { + it->moveItem(lastElement); + lastElementMap.insert(it->parent(),it); + } + else if ( TQString::localeAwareCompare(it->text(c),search_it->text(c)) < 0 ) { //we assume the list is sorted, as it should stay + if ( last_it ) it->moveItem(last_it); + break; + } + last_it = search_it; + } + } + } + } +} + +void DBListViewBase::removeElement( TQListViewItem *it, bool delete_item ) +{ + total--; + if ( !it ) return; + + TQListViewItem *lastElement = lastElementMap[it->parent()]; + if ( it == lastElement ) { + for ( TQListViewItem *search_it = (it->parent())?it->parent()->firstChild():firstChild(); search_it->nextSibling(); search_it = search_it->nextSibling() ) { + if ( it == search_it->nextSibling() ) { + lastElementMap.insert(it->parent(),search_it); + lastElement = search_it; + break; + } + } + + if ( lastElement == it || lastElement->rtti() == PREVLISTITEM_RTTI ) { //there are no more items in the view if this happens + if ( firstChild() && firstChild()->rtti() == PREVLISTITEM_RTTI ) { + activatePrev(); + it = 0; //keep 'delete it' below from segfault'ing + } + else if ( lastElement->nextSibling() && lastElement->nextSibling()->rtti() == NEXTLISTITEM_RTTI ) { + reload(); + it = 0; //keep 'delete it' below from segfault'ing + } + else //the list is now empty, there is no last element + lastElementMap.remove(it->parent()); + } + } + + if ( delete_item ) + delete it; +} + +bool DBListViewBase::handleElement( const TQString &name ) +{ + total++; + + TQListViewItem *lastElement = lastElementMap[0]; + + int c = 0;//FIXME: the column used should be variable (set by a subclass) + + int child_count = childCount(); + if ( child_count == 0 ) return true; + + if ( firstChild()->rtti() == PREVLISTITEM_RTTI || firstChild()->rtti() == 1006 ){ child_count--; } //"Prev" item + if ( child_count == 0 ) return true; + + if ( lastElement->nextSibling() ){ child_count--; } //"Next" item + + if ( curr_limit != -1 && child_count >= curr_limit ) { + TQListViewItem *firstElement = firstChild(); + if (firstElement->rtti() == PREVLISTITEM_RTTI || firstElement->rtti() == 1006 ) { + firstElement = firstElement->nextSibling(); + } + else if ( name < firstElement->text(c) ) { //provide access to this new element if we need to + new PrevListViewItem(this); + curr_offset++; + return false; + } + + if ( name < firstElement->text(c) ) { + curr_offset++; + return false; + } + else if ( name >= lastElement->text(c) ) { + if ( lastElement->nextSibling() == 0 ) + new NextListViewItem(this,lastElement); + + return false; + } + else { + removeElement(lastElement); + } + } + + return true; +} + +#include "dblistviewbase.moc" diff --git a/src/widgets/dblistviewbase.h b/src/widgets/dblistviewbase.h new file mode 100644 index 0000000..d1514fe --- /dev/null +++ b/src/widgets/dblistviewbase.h @@ -0,0 +1,89 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef LISTVIEWHANDLER_H +#define LISTVIEWHANDLER_H + +#include <tqobject.h> +#include <tqmap.h> + +#include <tdelistview.h> + +#define PREVLISTITEM_RTTI 1002 +#define NEXTLISTITEM_RTTI 1003 + +class KProgressDialog; + +class RecipeDB; + +enum ReloadFlags { + Load, /** Only performs the reload if the list hasn't already been loaded */ + ReloadIfPopulated, /** Only performs the reload if the list has been loaded */ + ForceReload /** Load/reload the list regardless of whether or not it's been loaded */ +}; + +class DBListViewBase : public TDEListView +{ +TQ_OBJECT + +public: + DBListViewBase( TQWidget *, RecipeDB *, int total ); + ~DBListViewBase(); + + void reload( ReloadFlags flag = Load ); + +signals: + void nextGroupLoaded(); + void prevGroupLoaded(); + +protected: + /** + * Called when the list view is ready to be used, i.e., it has been loaded with data. + * Until the list view has been loaded, we can ignore all database signals regarding changes + * of data. Therefore, subclasses should connect to these signals during this call. + */ + virtual void init(){} + virtual void load(int limit, int offset) = 0; + virtual void keyPressEvent( TQKeyEvent *e ); + bool handleElement( const TQString & ); + virtual void createElement( TQListViewItem * ); + void removeElement( TQListViewItem *, bool delete_item = true ); + + bool reloading(){ return bulk_load; } + void setSorting(int c){TDEListView::setSorting(c);} //don't do sorting, the database comes sorted from the database anyways + void setTotalItems(int); + + RecipeDB *database; + int curr_limit; + int curr_offset; + +protected slots: + void rename( TQListViewItem *, int c ); + void slotDoubleClicked( TQListViewItem * ); + +private: + void activatePrev(); + void activateNext(); + + //make this private because the data should always be synced with the database + void clear(){TDEListView::clear();} + + int total; + + bool bulk_load; + + TQMap<TQListViewItem*,TQListViewItem*> lastElementMap; + TQListViewItem *delete_me_later; + + KProgressDialog *m_progress; + int m_totalSteps; +}; + +#endif //LISTVIEWHANDLER_H diff --git a/src/widgets/fractioninput.cpp b/src/widgets/fractioninput.cpp new file mode 100644 index 0000000..68a3cfd --- /dev/null +++ b/src/widgets/fractioninput.cpp @@ -0,0 +1,121 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "fractioninput.h" + +#include <tqtimer.h> + +#include <tdeglobalsettings.h> + +#include "datablocks/ingredient.h" + +FractionInput::FractionInput( TQWidget *parent, MixedNumber::Format format ) : KLineEdit( parent ), + m_allowRange(false), + m_validateTimer(new TQTimer(this)), + m_format(format) +{ + setAlignment( TQt::AlignRight ); + + connect( this, TQ_SIGNAL(textChanged(const TQString&)), this, TQ_SLOT(slotStartValidateTimer()) ); + connect( m_validateTimer, TQ_SIGNAL(timeout()), this, TQ_SLOT(validate()) ); +} + +FractionInput::~FractionInput() +{ + delete m_validateTimer; +} + +void FractionInput::setValue( double d, double amount_offset ) +{ + MixedNumber m( d ); + setValue( m, amount_offset ); +} + +void FractionInput::setValue( const MixedNumber &m, double amount_offset ) +{ + TQString text = m.toString( m_format ); + if ( amount_offset > 0 ) { + text += "-" + MixedNumber(m+amount_offset).toString( MixedNumber::MixedNumberFormat ); + } + setText(text); +} + +void FractionInput::value( MixedNumber &amount, double &amount_offset ) const +{ + Ingredient i; i.setAmount( text() ); + + amount = MixedNumber(i.amount); + amount_offset = i.amount_offset; +} + +void FractionInput::value( double &amount, double &amount_offset ) const +{ + Ingredient i; i.setAmount( text() ); + + amount = i.amount; + amount_offset = i.amount_offset; +} + +MixedNumber FractionInput::value() const +{ + Ingredient i; i.setAmount( text() ); + + return MixedNumber(i.amount); +} + +MixedNumber FractionInput::minValue() const +{ + Ingredient i; i.setAmount( text() ); + + return MixedNumber(i.amount); +} + +MixedNumber FractionInput::maxValue() const +{ + Ingredient i; i.setAmount( text() ); + + return MixedNumber(i.amount_offset+i.amount); +} + +bool FractionInput::isInputValid() const +{ + if ( !m_allowRange && text().contains("-") ) + return false; + + bool ok; + Ingredient i; i.setAmount( text(), &ok ); + + return ok; +} + +void FractionInput::slotStartValidateTimer() +{ + if ( !m_validateTimer->isActive() ) + m_validateTimer->start( 1000, true ); + + if ( isInputValid() ) + emit valueChanged( value() ); +} + +void FractionInput::validate() +{ + if ( isInputValid() ) { + setPaletteForegroundColor( TDEGlobalSettings::textColor() ); + } + else + setPaletteForegroundColor( TQt::red ); +} + +bool FractionInput::isEmpty() const +{ + return text().isEmpty(); +} + +#include "fractioninput.moc" diff --git a/src/widgets/fractioninput.h b/src/widgets/fractioninput.h new file mode 100644 index 0000000..b682a32 --- /dev/null +++ b/src/widgets/fractioninput.h @@ -0,0 +1,62 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef FRACTIONINPUT_H +#define FRACTIONINPUT_H + +#include <klineedit.h> + +#include "datablocks/mixednumber.h" + +class TQTimer; + +/** A KLineEdit widget extended to allow input of decimals and fractions or ranges of such. + * Input is returned as a @ref MixedNumber class. + * @author Jason Kivlighn + */ +class FractionInput : public KLineEdit +{ +TQ_OBJECT + +public: + FractionInput( TQWidget *parent = 0, MixedNumber::Format = MixedNumber::MixedNumberFormat ); + ~FractionInput(); + + void setAllowRange( bool b ){ m_allowRange = b; } + + void setValue( double amount, double amount_offset ); + void setValue( const MixedNumber &, double amount_offset ); + + void value( MixedNumber &amount, double &amount_offset ) const; + void value( double &amount, double &amount_offset ) const; + MixedNumber minValue() const; + MixedNumber maxValue() const; + MixedNumber value() const; + + bool isInputValid() const; + bool isEmpty() const; + +signals: + void valueChanged( const MixedNumber & ); + +public slots: + void validate(); + +private slots: + void slotStartValidateTimer(); + +private: + bool m_allowRange; + TQTimer *m_validateTimer; + MixedNumber::Format m_format; +}; + +#endif //FRACTIONINPUT_H + diff --git a/src/widgets/headercombobox.cpp b/src/widgets/headercombobox.cpp new file mode 100644 index 0000000..3c43cbc --- /dev/null +++ b/src/widgets/headercombobox.cpp @@ -0,0 +1,42 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "headercombobox.h" + +#include <tqlistbox.h> + +#include "backends/recipedb.h" +#include "datablocks/elementlist.h" + +HeaderComboBox::HeaderComboBox( bool b, TQWidget *parent, RecipeDB *db ) : KComboBox( b, parent ), + database( db ) +{ +} + +void HeaderComboBox::reload() +{ + TQString remember_text = currentText(); + + ElementList headerList; + database->loadIngredientGroups( &headerList ); + + clear(); + + for ( ElementList::const_iterator it = headerList.begin(); it != headerList.end(); ++it ) { + insertItem((*it).name); + completionObject()->addItem((*it).name); + } + + if ( listBox()->findItem( remember_text, TQt::ExactMatch ) ) { + setCurrentText( remember_text ); + } +} + +#include "headercombobox.moc" diff --git a/src/widgets/headercombobox.h b/src/widgets/headercombobox.h new file mode 100644 index 0000000..5e07b20 --- /dev/null +++ b/src/widgets/headercombobox.h @@ -0,0 +1,34 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef HEADERCOMBOBOX_H +#define HEADERCOMBOBOX_H + +#include <kcombobox.h> + +#include "datablocks/element.h" + +class RecipeDB; + +class HeaderComboBox : public KComboBox +{ + TQ_OBJECT + +public: + HeaderComboBox( bool, TQWidget *parent, RecipeDB *db ); + + void reload(); + +private: + RecipeDB *database; +}; + +#endif //HEADERCOMBOBOX_H + diff --git a/src/widgets/headerlistview.cpp b/src/widgets/headerlistview.cpp new file mode 100644 index 0000000..51e9d4c --- /dev/null +++ b/src/widgets/headerlistview.cpp @@ -0,0 +1,196 @@ +/*************************************************************************** +* Copyright (C) 2004 by * +* Jason Kivlighn ([email protected]) * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "headerlistview.h" + +#include <tdemessagebox.h> +#include <tdeconfig.h> +#include <tdelocale.h> +#include <tdeglobal.h> +#include <kiconloader.h> +#include <tdepopupmenu.h> + +#include "backends/recipedb.h" +#include "dialogs/createelementdialog.h" +#include "dialogs/dependanciesdialog.h" + +HeaderListView::HeaderListView( TQWidget *parent, RecipeDB *db ) : DBListViewBase( parent,db,db->unitCount() ) +{ + setAllColumnsShowFocus( true ); + setDefaultRenameAction( TQListView::Reject ); +} + +void HeaderListView::init() +{ + connect( database, TQ_SIGNAL( ingGroupCreated( const Element & ) ), TQ_SLOT( checkCreateHeader( const Element & ) ) ); + connect( database, TQ_SIGNAL( ingGroupRemoved( int ) ), TQ_SLOT( removeHeader( int ) ) ); +} + +void HeaderListView::load( int /*limit*/, int /*offset*/ ) +{ + ElementList headerList; + database->loadIngredientGroups( &headerList ); + + setTotalItems(headerList.count()); + + for ( ElementList::const_iterator it = headerList.begin(); it != headerList.end(); ++it ) { + createHeader( *it ); + } +} + +void HeaderListView::checkCreateHeader( const Element &el ) +{ + if ( handleElement(el.name) ) { //only create this header if the base class okays it + createHeader(el); + } +} + + +StdHeaderListView::StdHeaderListView( TQWidget *parent, RecipeDB *db, bool editable ) : HeaderListView( parent, db ) +{ + addColumn( i18n( "Header" ) ); + + TDEConfig * config = TDEGlobal::config(); + config->setGroup( "Advanced" ); + bool show_id = config->readBoolEntry( "ShowID", false ); + addColumn( i18n( "Id" ), show_id ? -1 : 0 ); + + if ( editable ) { + setRenameable( 0, true ); + + TDEIconLoader *il = new TDEIconLoader; + + kpop = new TDEPopupMenu( this ); + kpop->insertItem( il->loadIcon( "document-new", TDEIcon::NoGroup, 16 ), i18n( "&Create" ), this, TQ_SLOT( createNew() ), CTRL + Key_C ); + kpop->insertItem( il->loadIcon( "edit-delete", TDEIcon::NoGroup, 16 ), i18n( "&Delete" ), this, TQ_SLOT( remove + () ), Key_Delete ); + kpop->insertItem( il->loadIcon( "edit", TDEIcon::NoGroup, 16 ), i18n( "&Rename" ), this, TQ_SLOT( rename() ), CTRL + Key_R ); + kpop->polish(); + + delete il; + + connect( this, TQ_SIGNAL( contextMenu( TDEListView *, TQListViewItem *, const TQPoint & ) ), TQ_SLOT( showPopup( TDEListView *, TQListViewItem *, const TQPoint & ) ) ); + connect( this, TQ_SIGNAL( doubleClicked( TQListViewItem*, const TQPoint &, int ) ), this, TQ_SLOT( modHeader( TQListViewItem*, const TQPoint &, int ) ) ); + connect( this, TQ_SIGNAL( itemRenamed( TQListViewItem*, const TQString &, int ) ), this, TQ_SLOT( saveHeader( TQListViewItem*, const TQString &, int ) ) ); + } +} + +void StdHeaderListView::showPopup( TDEListView * /*l*/, TQListViewItem *i, const TQPoint &p ) +{ + if ( i ) + kpop->exec( p ); +} + +void StdHeaderListView::createNew() +{ + CreateElementDialog * headerDialog = new CreateElementDialog( this, i18n("Header") ); + + if ( headerDialog->exec() == TQDialog::Accepted ) { + TQString result = headerDialog->newElementName(); + + //check bounds first + if ( checkBounds( result ) ) + database->createNewIngGroup( result ); + } + delete headerDialog; +} + +void StdHeaderListView::remove() +{ + // Find selected header item + TQListViewItem * it = selectedItem(); + + if ( it ) { + int headerID = it->text( 1 ).toInt(); + + ElementList recipeDependancies; + database->findUseOfIngGroupInRecipes( &recipeDependancies, headerID ); + + if ( recipeDependancies.isEmpty() ) + database->removeIngredientGroup( headerID ); + else { // need warning! + ListInfo info; + info.list = recipeDependancies; + info.name = i18n( "Recipes" ); + + DependanciesDialog warnDialog( this, info ); + warnDialog.setCustomWarning( i18n("You are about to permanantly delete recipes from your database.") ); + if ( warnDialog.exec() == TQDialog::Accepted ) + database->removeIngredientGroup( headerID ); + } + } +} + +void StdHeaderListView::rename() +{ + TQListViewItem * item = currentItem(); + + if ( item ) + HeaderListView::rename( item, 0 ); +} + +void StdHeaderListView::createHeader( const Element &header ) +{ + createElement(new TQListViewItem( this, header.name, TQString::number( header.id ) )); +} + +void StdHeaderListView::removeHeader( int id ) +{ + TQListViewItem * item = findItem( TQString::number( id ), 1 ); + removeElement(item); +} + +void StdHeaderListView::modHeader( TQListViewItem* i, const TQPoint & /*p*/, int c ) +{ + if ( i ) + HeaderListView::rename( i, c ); +} + +void StdHeaderListView::saveHeader( TQListViewItem* i, const TQString &text, int /*c*/ ) +{ + if ( !checkBounds( text ) ) { + reload(ForceReload); //reset the changed text + return ; + } + + int existing_id = database->findExistingIngredientGroupByName( text ); + int header_id = i->text( 1 ).toInt(); + if ( existing_id != -1 && existing_id != header_id ) { //already exists with this label... merge the two + switch ( KMessageBox::warningContinueCancel( this, i18n( "This header already exists. Continuing will merge these two headers into one. Are you sure?" ) ) ) { + case KMessageBox::Continue: { + database->modIngredientGroup( header_id, i->text( 0 ) ); + database->mergeIngredientGroups( header_id, existing_id ); + break; + } + default: + reload(ForceReload); + break; + } + } + else { + database->modIngredientGroup( header_id, i->text( 0 ) ); + } +} + +bool StdHeaderListView::checkBounds( const TQString &header ) +{ + if ( header.length() > uint(database->maxIngGroupNameLength()) ) { + KMessageBox::error( this, TQString( i18n( "Header cannot be longer than %1 characters." ) ).arg( database->maxIngGroupNameLength() ) ); + return false; + } + else if ( header.stripWhiteSpace().isEmpty() ) + return false; + + return true; +} + +#include "headerlistview.moc" diff --git a/src/widgets/headerlistview.h b/src/widgets/headerlistview.h new file mode 100644 index 0000000..716d002 --- /dev/null +++ b/src/widgets/headerlistview.h @@ -0,0 +1,70 @@ +/*************************************************************************** +* Copyright (C) 2004 by * +* Jason Kivlighn ([email protected]) * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef HEADERLISTVIEW_H +#define HEADERLISTVIEW_H + +#include "dblistviewbase.h" + +#include "datablocks/element.h" + +class RecipeDB; +class TDEPopupMenu; + +class HeaderListView : public DBListViewBase +{ + TQ_OBJECT + +public: + HeaderListView( TQWidget *parent, RecipeDB *db ); + +public slots: + virtual void load( int curr_limit, int curr_offset ); + +protected slots: + virtual void createHeader( const Element & ) = 0; + virtual void removeHeader( int ) = 0; + + void checkCreateHeader( const Element &el ); + +protected: + virtual void init(); +}; + +class StdHeaderListView : public HeaderListView +{ + TQ_OBJECT + +public: + StdHeaderListView( TQWidget *parent, RecipeDB *db, bool editable = false ); + +protected: + virtual void createHeader( const Element & ); + virtual void removeHeader( int ); + +private slots: + void showPopup( TDEListView *, TQListViewItem *, const TQPoint & ); + + void createNew(); + void remove(); + void rename(); + + void modHeader( TQListViewItem* i, const TQPoint &p, int c ); + void saveHeader( TQListViewItem* i, const TQString &text, int c ); + +private: + bool checkBounds( const TQString &unit ); + + TDEPopupMenu *kpop; +}; + +#endif //HEADERLISTVIEW_H diff --git a/src/widgets/inglistviewitem.cpp b/src/widgets/inglistviewitem.cpp new file mode 100644 index 0000000..0670dc4 --- /dev/null +++ b/src/widgets/inglistviewitem.cpp @@ -0,0 +1,225 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "inglistviewitem.h" + +#include <tdelocale.h> +#include <tdeconfig.h> +#include <tdeglobal.h> +#include <kdebug.h> + +#include "datablocks/unit.h" +#include "datablocks/mixednumber.h" + +IngSubListViewItem::IngSubListViewItem( TQListViewItem* qli, const Ingredient &i ) : IngListViewItem( qli, 0, i ) +{ +} + +TQString IngSubListViewItem::text( int column ) const +{ + if ( column == 0 ) { + //kdDebug()<<"displaying col 0 for "<<m_ing.name<<endl; + return TQString("%1 ").arg(i18n("OR"))+m_ing.name; + //return m_ing.name; + } + else + return IngListViewItem::text(column); + +} + +void IngSubListViewItem::setText( int column, const TQString &text ) +{ + switch ( column ) { + case 0: { + TQString compare = TQString("%1 ").arg(i18n("OR")); + if ( text.left(compare.length()) == compare ) + m_ing.name = text.right(text.length()-compare.length()); + else + m_ing.name = text; + break; + } + default: + IngListViewItem::setText(column,text); + break; + } +} + +int IngSubListViewItem::rtti() const +{ + return INGSUBLISTVIEWITEM_RTTI; +} + + +IngListViewItem::IngListViewItem( TQListView* qlv, const Ingredient &i ) : TQListViewItem( qlv ) +{ + init( i ); +} + +IngListViewItem::IngListViewItem( TQListView* qlv, TQListViewItem *after, const Ingredient &i ) : TQListViewItem( qlv, after ) +{ + init( i ); +} + +IngListViewItem::IngListViewItem( TQListViewItem* qli, TQListViewItem *after, const Ingredient &i ) : TQListViewItem( qli, after ) +{ + init( i ); +} + +int IngListViewItem::rtti() const +{ + return INGLISTVIEWITEM_RTTI; +} + +Ingredient IngListViewItem::ingredient() const +{ + return m_ing; +} + +void IngListViewItem::setAmount( double amount, double amount_offset ) +{ + amount_str = TQString::null; + + if ( amount > 0 ) { + TDEConfig * config = TDEGlobal::config(); + config->setGroup( "Formatting" ); + + if ( config->readBoolEntry( "Fraction" ) ) + amount_str = MixedNumber( amount ).toString(); + else + amount_str = beautify( TDEGlobal::locale() ->formatNumber( amount, 5 ) ); + } + if ( amount_offset > 0 ) { + if ( amount < 1e-10 ) + amount_str += "0"; + amount_str += "-" + MixedNumber(amount+amount_offset).toString(); + } + + m_ing.amount = amount; + m_ing.amount_offset = amount_offset; + + //FIXME: make sure the right unit is showing after changing this (force a repaint... repaint() doesn't do the job right because it gets caught in a loop) +} + +void IngListViewItem::setUnit( const Unit &unit ) +{ + //### This shouldn't be necessary... the db backend should ensure this doesn't happen + if ( !unit.name.isEmpty() ) + m_ing.units.name = unit.name; + if ( !unit.plural.isEmpty() ) + m_ing.units.plural = unit.plural; +} + +void IngListViewItem::setPrepMethod( const TQString &prepMethod ) +{ + m_ing.prepMethodList = ElementList::split(",",prepMethod); +} + +void IngListViewItem::setText( int column, const TQString &text ) +{ + switch ( column ) { + case 0: { + m_ing.name = text; + break; + } + case 1: { + Ingredient i; i.setAmount(text); + setAmount( i.amount, i.amount_offset ); + break; + } + case 2: + setUnit( Unit( text, m_ing.amount+m_ing.amount_offset ) ); + break; + case 3: + setPrepMethod( text ); + break; + default: + break; + } +} + +TQString IngListViewItem::text( int column ) const +{ + switch ( column ) { + case 0: + return m_ing.name; + break; + case 1: + return amount_str; + break; + case 2: + return ( m_ing.amount+m_ing.amount_offset > 1 ) ? m_ing.units.plural : m_ing.units.name; + break; + case 3: + return m_ing.prepMethodList.join(","); + break; + default: + return ( TQString::null ); + } +} + +void IngListViewItem::init( const Ingredient &i ) +{ + m_ing = i; + + setAmount( i.amount, i.amount_offset ); +} + + +IngGrpListViewItem::IngGrpListViewItem( TQListView* qlv, TQListViewItem *after, const TQString &group, int id ) : TQListViewItem( qlv, after ) +{ + init( group, id ); +} + +int IngGrpListViewItem::rtti() const +{ + return INGGRPLISTVIEWITEM_RTTI; +} + +TQString IngGrpListViewItem::group() const +{ + return m_group; +} + +int IngGrpListViewItem::id() const +{ + return m_id; +} + +TQString IngGrpListViewItem::text( int column ) const +{ + switch ( column ) { + case 0: + return m_group + ":"; + break; + default: + return ( TQString::null ); + } +} + +void IngGrpListViewItem::setText( int column, const TQString &text ) +{ + switch ( column ) { + case 0: + if ( text.right(1) == ":" ) + m_group = text.left(text.length()-1); + else + m_group = text; + break; + default: + break; + } +} + +void IngGrpListViewItem::init( const TQString &group, int id ) +{ + m_group = group; + m_id = id; +} + diff --git a/src/widgets/inglistviewitem.h b/src/widgets/inglistviewitem.h new file mode 100644 index 0000000..89de283 --- /dev/null +++ b/src/widgets/inglistviewitem.h @@ -0,0 +1,82 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef INGLISTVIEWITEM_H +#define INGLISTVIEWITEM_H + +#include "tqlistview.h" + +#include "datablocks/ingredient.h" + +#define INGGRPLISTVIEWITEM_RTTI 1003 +#define INGLISTVIEWITEM_RTTI 1004 +#define INGSUBLISTVIEWITEM_RTTI 1009 + +class IngListViewItem : public TQListViewItem +{ +public: + IngListViewItem( TQListView* qlv, const Ingredient &i ); + IngListViewItem( TQListView* qlv, TQListViewItem *after, const Ingredient &i ); + IngListViewItem( TQListViewItem* qli, TQListViewItem *after, const Ingredient &i ); + + int rtti() const; + + Ingredient ingredient() const; + + void setAmount( double amount, double amount_offset ); + void setUnit( const Unit &unit ); + void setPrepMethod( const TQString &prepMethod ); + +protected: + Ingredient m_ing; + TQString amount_str; + +public: + virtual TQString text( int column ) const; + virtual void setText( int column, const TQString &text ); + +private: + void init( const Ingredient &i ); +}; + + +class IngSubListViewItem : public IngListViewItem +{ +public: + IngSubListViewItem( TQListViewItem* qli, const Ingredient &i ); + + virtual TQString text( int column ) const; + virtual void setText( int column, const TQString &text ); + virtual int rtti() const; +}; + + +class IngGrpListViewItem : public TQListViewItem +{ +public: + IngGrpListViewItem( TQListView* qlv, TQListViewItem *after, const TQString &group, int id ); + + int rtti() const; + + TQString group() const; + int id() const; + + virtual TQString text( int column ) const; + virtual void setText( int column, const TQString &text ); + +protected: + TQString m_group; + int m_id; + +private: + void init( const TQString &group, int id ); +}; + +#endif diff --git a/src/widgets/ingredientcombobox.cpp b/src/widgets/ingredientcombobox.cpp new file mode 100644 index 0000000..0e036b0 --- /dev/null +++ b/src/widgets/ingredientcombobox.cpp @@ -0,0 +1,188 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "ingredientcombobox.h" + +#include <tqlistbox.h> +#include <tqtimer.h> + +#include <kdebug.h> +#include <tdeapplication.h> +#include <tdeglobal.h> +#include <tdeconfig.h> + +#include "backends/recipedb.h" +#include "datablocks/elementlist.h" + +IngredientComboBox::IngredientComboBox( bool b, TQWidget *parent, RecipeDB *db, const TQString &specialItem ) : KComboBox( b, parent ), + database( db ), loading_at(0), load_timer(new TQTimer(this)), m_specialItem(specialItem) +{ + connect( load_timer, TQ_SIGNAL(timeout()), TQ_SLOT(loadMore()) ); + completionObject()->setIgnoreCase(true); +} + +void IngredientComboBox::reload() +{ + TQString remember_text; + if ( editable() ) + remember_text = lineEdit()->text(); + + ElementList ingredientList; + database->loadIngredients( &ingredientList ); + + clear(); + ingredientComboRows.clear(); + + int row = 0; + if ( !m_specialItem.isNull() ) { + insertItem(m_specialItem); + ingredientComboRows.insert( row, -1 ); + row++; + } + for ( ElementList::const_iterator it = ingredientList.begin(); it != ingredientList.end(); ++it, ++row ) { + insertItem((*it).name); + completionObject()->addItem((*it).name); + ingredientComboRows.insert( row, (*it).id ); + } + + if ( editable() ) + setEditText( remember_text ); + + database->disconnect( this ); + connect( database, TQ_SIGNAL( ingredientCreated( const Element & ) ), TQ_SLOT( createIngredient( const Element & ) ) ); + connect( database, TQ_SIGNAL( ingredientRemoved( int ) ), TQ_SLOT( removeIngredient( int ) ) ); +} + +void IngredientComboBox::loadMore() +{ + if ( loading_at >= ing_count-1 ) { + endLoad(); + return; + } + + ElementList ingredientList; + database->loadIngredients( &ingredientList, load_limit, loading_at ); + + for ( ElementList::const_iterator it = ingredientList.begin(); it != ingredientList.end(); ++it, ++loading_at ) { + insertItem((*it).name); + completionObject()->addItem((*it).name); + ingredientComboRows.insert( loading_at, (*it).id ); + } +} + +void IngredientComboBox::startLoad() +{ + //don't receive ingredient created/removed events from the database + database->disconnect( this ); + + TDEConfig * config = TDEGlobal::config(); config->setGroup( "Performance" ); + load_limit = config->readNumEntry( "Limit", -1 ); + if ( load_limit == -1 ) { + reload(); + endLoad(); + } + else { + loading_at = 0; + ing_count = database->ingredientCount(); + + load_timer->start( 0, false ); + } +} + +void IngredientComboBox::endLoad() +{ + load_timer->stop(); + + //now we're ready to receive ingredient created/removed events from the database + connect( database, TQ_SIGNAL( ingredientCreated( const Element & ) ), TQ_SLOT( createIngredient( const Element & ) ) ); + connect( database, TQ_SIGNAL( ingredientRemoved( int ) ), TQ_SLOT( removeIngredient( int ) ) ); +} + +int IngredientComboBox::id( int row ) +{ + return ingredientComboRows[ row ]; +} + +int IngredientComboBox::id( const TQString &ing ) +{ + for ( int i = 0; i < count(); i++ ) { + if ( ing == text( i ) ) + return id(i); + } + kdDebug()<<"Warning: couldn't find the ID for "<<ing<<endl; + return -1; +} + +void IngredientComboBox::createIngredient( const Element &element ) +{ + int row = findInsertionPoint( element.name ); + + TQString remember_text; + if ( editable() ) + remember_text = lineEdit()->text(); + + insertItem( element.name, row ); + completionObject()->addItem(element.name); + + if ( editable() ) + lineEdit()->setText( remember_text ); + + //now update the map by pushing everything after this item down + TQMap<int, int> new_map; + for ( TQMap<int, int>::iterator it = ingredientComboRows.begin(); it != ingredientComboRows.end(); ++it ) { + if ( it.key() >= row ) { + new_map.insert( it.key() + 1, it.data() ); + } + else + new_map.insert( it.key(), it.data() ); + } + ingredientComboRows = new_map; + ingredientComboRows.insert( row, element.id ); +} + +void IngredientComboBox::removeIngredient( int id ) +{ + int row = -1; + for ( TQMap<int, int>::iterator it = ingredientComboRows.begin(); it != ingredientComboRows.end(); ++it ) { + if ( it.data() == id ) { + row = it.key(); + completionObject()->removeItem( text(row) ); + removeItem( row ); + ingredientComboRows.remove( it ); + break; + } + } + + if ( row == -1 ) + return ; + + //now update the map by pushing everything after this item up + TQMap<int, int> new_map; + for ( TQMap<int, int>::iterator it = ingredientComboRows.begin(); it != ingredientComboRows.end(); ++it ) { + if ( it.key() > row ) { + new_map.insert( it.key() - 1, it.data() ); + } + else + new_map.insert( it.key(), it.data() ); + } + ingredientComboRows = new_map; +} + +int IngredientComboBox::findInsertionPoint( const TQString &name ) +{ + for ( int i = 0; i < count(); i++ ) { + if ( TQString::localeAwareCompare( name, text( i ) ) < 0 ) + return i; + } + + return count(); +} + +#include "ingredientcombobox.moc" diff --git a/src/widgets/ingredientcombobox.h b/src/widgets/ingredientcombobox.h new file mode 100644 index 0000000..9bd5fb7 --- /dev/null +++ b/src/widgets/ingredientcombobox.h @@ -0,0 +1,58 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef INGREDIENTCOMBOBOX_H +#define INGREDIENTCOMBOBOX_H + +#include <kcombobox.h> + +#include <tqmap.h> + +#include "datablocks/element.h" + +class TQTimer; + +class RecipeDB; +class ElementList; + +class IngredientComboBox : public KComboBox +{ + TQ_OBJECT + +public: + IngredientComboBox( bool, TQWidget *parent, RecipeDB *db, const TQString &specialItem = TQString::null ); + + void reload(); + int id( int row ); + int id( const TQString &ing ); + + void startLoad(); + void endLoad(); + +private slots: + void createIngredient( const Element &element ); + void removeIngredient( int id ); + + int findInsertionPoint( const TQString &name ); + void loadMore(); + +private: + RecipeDB *database; + TQMap<int, int> ingredientComboRows; // Contains the category id for every given row in the category combobox + + int loading_at; + int ing_count; + int load_limit; + TQTimer *load_timer; + TQString m_specialItem; +}; + +#endif //INGREDIENTCOMBOBOX_H + diff --git a/src/widgets/ingredientinputwidget.cpp b/src/widgets/ingredientinputwidget.cpp new file mode 100644 index 0000000..3c0aa7e --- /dev/null +++ b/src/widgets/ingredientinputwidget.cpp @@ -0,0 +1,542 @@ +/*************************************************************************** +* Copyright (C) 2003-2005 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* Copyright (C) 2006 Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "ingredientinputwidget.h" + +#include <tqlabel.h> +#include <tqwidgetstack.h> +#include <tqhbox.h> +#include <tqvbox.h> +#include <tqgroupbox.h> +#include <tqbuttongroup.h> +#include <tqradiobutton.h> +#include <tqcheckbox.h> + +#include <kcombobox.h> +#include <tdelocale.h> +#include <tdemessagebox.h> +#include <kdebug.h> + +#include "backends/recipedb.h" +#include "datablocks/unit.h" +#include "widgets/fractioninput.h" +#include "widgets/ingredientcombobox.h" +#include "widgets/headercombobox.h" +#include "widgets/prepmethodcombobox.h" +#include "dialogs/createunitdialog.h" + +#include "profiling.h" + +IngredientInput::IngredientInput( RecipeDB *db, TQWidget *parent, bool allowHeader ) : TQHBox(parent), database(db) +{ + TQVBox *ingredientVBox = new TQVBox( this ); + TQHBox *typeHBox = new TQHBox( ingredientVBox ); + + if ( allowHeader ) { + typeButtonGrp = new TQButtonGroup(); + TQRadioButton *ingredientRadioButton = new TQRadioButton( i18n( "Ingredient:" ), typeHBox ); + typeButtonGrp->insert( ingredientRadioButton, 0 ); + + TQRadioButton *headerRadioButton = new TQRadioButton( i18n( "Ingredient grouping name", "Header:" ), typeHBox ); + typeButtonGrp->insert( headerRadioButton, 1 ); + + typeButtonGrp->setButton( 0 ); + connect( typeButtonGrp, TQ_SIGNAL( clicked( int ) ), TQ_SLOT( typeButtonClicked( int ) ) ); + } + else { + (void) new TQLabel( i18n( "Ingredient:" ), typeHBox ); + typeButtonGrp = 0; + } + + header_ing_stack = new TQWidgetStack(ingredientVBox); + ingredientBox = new IngredientComboBox( TRUE, header_ing_stack, database ); + ingredientBox->setAutoCompletion( TRUE ); + ingredientBox->lineEdit() ->disconnect( ingredientBox ); //so hitting enter doesn't enter the item into the box + ingredientBox->setSizePolicy( TQSizePolicy( TQSizePolicy::Ignored, TQSizePolicy::Fixed ) ); + header_ing_stack->addWidget( ingredientBox ); + headerBox = new HeaderComboBox( TRUE, header_ing_stack, database ); + headerBox->setAutoCompletion( TRUE ); + headerBox->lineEdit() ->disconnect( ingredientBox ); //so hitting enter doesn't enter the item into the box + headerBox->setSizePolicy( TQSizePolicy( TQSizePolicy::Ignored, TQSizePolicy::Fixed ) ); + header_ing_stack->addWidget( headerBox ); + + TQVBox *amountVBox = new TQVBox( this ); + amountLabel = new TQLabel( i18n( "Amount:" ), amountVBox ); + amountEdit = new FractionInput( amountVBox ); + amountEdit->setAllowRange(true); + amountEdit->setSizePolicy( TQSizePolicy( TQSizePolicy::Minimum, TQSizePolicy::Fixed ) ); + + TQVBox *unitVBox = new TQVBox( this ); + unitLabel = new TQLabel( i18n( "Unit:" ), unitVBox ); + unitBox = new KComboBox( TRUE, unitVBox ); + unitBox->setAutoCompletion( TRUE ); + unitBox->lineEdit() ->disconnect( unitBox ); //so hitting enter doesn't enter the item into the box + unitBox->setSizePolicy( TQSizePolicy( TQSizePolicy::Ignored, TQSizePolicy::Fixed ) ); + + TQVBox *prepMethodVBox = new TQVBox( this ); + prepMethodLabel = new TQLabel( i18n( "Preparation Method:" ), prepMethodVBox ); + prepMethodBox = new PrepMethodComboBox( TRUE, prepMethodVBox, database ); + prepMethodBox->setAutoCompletion( TRUE ); + prepMethodBox->lineEdit() ->disconnect( prepMethodBox ); //so hitting enter doesn't enter the item into the box + prepMethodBox->setSizePolicy( TQSizePolicy( TQSizePolicy::Ignored, TQSizePolicy::Fixed ) ); + + orButton = new TQCheckBox( i18n( "OR" ), this ); + + setStretchFactor( ingredientVBox, 5 ); + setStretchFactor( amountVBox, 1 ); + setStretchFactor( unitVBox, 2 ); + setStretchFactor( prepMethodVBox, 3 ); + + connect( ingredientBox, TQ_SIGNAL( activated( int ) ), this, TQ_SLOT( loadUnitListCombo() ) ); + connect( ingredientBox->lineEdit(), TQ_SIGNAL( lostFocus() ), this, TQ_SLOT( slotIngredientBoxLostFocus() ) ); + connect( unitBox->lineEdit(), TQ_SIGNAL( lostFocus() ), this, TQ_SLOT( slotUnitBoxLostFocus() ) ); + connect( prepMethodBox->lineEdit(), TQ_SIGNAL( lostFocus() ), this, TQ_SLOT( slotPrepMethodBoxLostFocus() ) ); + connect( orButton, TQ_SIGNAL( toggled(bool) ), this, TQ_SLOT( orToggled(bool) ) ); + + connect( unitBox->lineEdit(), TQ_SIGNAL( returnPressed() ), this, TQ_SLOT( signalIngredient() ) ); + connect( ingredientBox->lineEdit(), TQ_SIGNAL( returnPressed() ), this, TQ_SLOT( signalIngredient() ) ); + connect( headerBox->lineEdit(), TQ_SIGNAL( returnPressed() ), this, TQ_SLOT( signalIngredient() ) ); + connect( prepMethodBox->lineEdit(), TQ_SIGNAL( returnPressed() ), this, TQ_SLOT( signalIngredient() ) ); + connect( amountEdit, TQ_SIGNAL( returnPressed( const TQString & ) ), this, TQ_SLOT( signalIngredient() ) ); + + unitComboList = new UnitList; + + setFocusProxy( ingredientBox ); +} + +IngredientInput::~IngredientInput() +{ + delete unitComboList; + delete typeButtonGrp; +} + +void IngredientInput::clear() +{ + unitComboList->clear(); + + orButton->setChecked(false); + typeButtonGrp->setButton( 0 ); //put back to ingredient input + typeButtonClicked( 0 ); + + amountEdit->clear(); + ingredientBox->lineEdit()->setText(""); + prepMethodBox->lineEdit()->setText(""); + headerBox->lineEdit()->setText(""); + unitBox->lineEdit()->setText(""); +} + +void IngredientInput::orToggled(bool b) +{ + emit orToggled(b,this); +} + +void IngredientInput::reloadCombos() +{ + //these only needed to be loaded once + if ( ingredientBox->count() == 0 ) { + START_TIMER("Loading ingredient input auto-completion"); + ingredientBox->reload(); + END_TIMER(); + } + if ( headerBox->count() == 0 ) { + START_TIMER("Loading ingredient header input auto-completion"); + headerBox->reload(); + END_TIMER(); + } + if ( prepMethodBox->count() == 0 ) { + START_TIMER("Loading prep method input auto-completion"); + prepMethodBox->reload(); + END_TIMER(); + } + + loadUnitListCombo(); +} + +void IngredientInput::slotIngredientBoxLostFocus( void ) +{ + if ( ingredientBox->contains( ingredientBox->currentText() ) ) { + ingredientBox->setCurrentItem( ingredientBox->currentText() ); + loadUnitListCombo(); + } + else { + unitBox->clear(); + unitBox->completionObject() ->clear(); + unitComboList->clear(); + } +} + +void IngredientInput::slotUnitBoxLostFocus() +{ + if ( unitBox->contains( unitBox->currentText() ) ) + unitBox->setCurrentItem( unitBox->currentText() ); +} + +void IngredientInput::slotPrepMethodBoxLostFocus() +{ + if ( prepMethodBox->contains( prepMethodBox->currentText() ) ) + prepMethodBox->setCurrentItem( prepMethodBox->currentText() ); +} + +void IngredientInput::typeButtonClicked( int button_id ) +{ + if ( amountEdit->isEnabled() == !bool( button_id ) ) //it is already set (the same button was clicked more than once) + return ; + + amountEdit->setEnabled( !bool( button_id ) ); + unitBox->setEnabled( !bool( button_id ) ); + prepMethodBox->setEnabled( !bool( button_id ) ); + + if ( button_id == 1 ) { //Header + header_ing_stack->raiseWidget( headerBox ); + } + else { + header_ing_stack->raiseWidget( ingredientBox ); + } +} + +void IngredientInput::enableHeader( bool enable ) +{ + if ( !enable ) { + typeButtonGrp->setButton( 0 ); //put back to ingredient input + typeButtonClicked( 0 ); + } + typeButtonGrp->find(1)->setEnabled(enable); +} + +void IngredientInput::signalIngredient() +{ + //validate input; if successful, emit signal + if ( isHeader() ) { + if ( header().isEmpty() ) + return; + } + else { + if ( !isInputValid() ) + return; + } + + emit addIngredient(); +} + +bool IngredientInput::isInputValid() +{ + if ( ingredientBox->currentText().stripWhiteSpace().isEmpty() ) { + KMessageBox::error( this, i18n( "Please enter an ingredient" ), TQString::null ); + ingredientBox->setFocus(); + return false; + } + return checkAmountEdit() && checkBounds(); +} + +bool IngredientInput::checkBounds() +{ + if ( ingredientBox->currentText().length() > uint(database->maxIngredientNameLength()) ) { + KMessageBox::error( this, TQString( i18n( "Ingredient name cannot be longer than %1 characters." ) ).arg( database->maxIngredientNameLength() ) ); + ingredientBox->lineEdit() ->setFocus(); + ingredientBox->lineEdit() ->selectAll(); + return false; + } + + if ( unitBox->currentText().length() > uint(database->maxUnitNameLength()) ) { + KMessageBox::error( this, TQString( i18n( "Unit name cannot be longer than %1 characters." ) ).arg( database->maxUnitNameLength() ) ); + unitBox->lineEdit() ->setFocus(); + unitBox->lineEdit() ->selectAll(); + return false; + } + + TQStringList prepMethodList = TQStringList::split(",",prepMethodBox->currentText()); + for ( TQStringList::const_iterator it = prepMethodList.begin(); it != prepMethodList.end(); ++it ) { + if ( (*it).stripWhiteSpace().length() > uint(database->maxPrepMethodNameLength()) ) + { + KMessageBox::error( this, TQString( i18n( "Preparation method cannot be longer than %1 characters." ) ).arg( database->maxPrepMethodNameLength() ) ); + prepMethodBox->lineEdit() ->setFocus(); + prepMethodBox->lineEdit() ->selectAll(); + return false; + } + } + + return true; +} + +bool IngredientInput::checkAmountEdit() +{ + if ( amountEdit->isInputValid() ) + return true; + else { + KMessageBox::error( this, i18n( "Amount field contains invalid input." ), + i18n( "Invalid input" ) ); + amountEdit->setFocus(); + amountEdit->selectAll(); + return false; + } +} + +void IngredientInput::loadUnitListCombo() +{ + TQString store_unit = unitBox->currentText(); + unitBox->clear(); // Empty the combo first + unitBox->completionObject() ->clear(); + + int comboIndex = ingredientBox->currentItem(); + int comboCount = ingredientBox->count(); + + if ( comboCount > 0 ) { // If not, the list may be empty (no ingredient list defined) and crashes while reading + int selectedIngredient = ingredientBox->id( comboIndex ); + database->loadPossibleUnits( selectedIngredient, unitComboList ); + + //Populate this data into the ComboBox + for ( UnitList::const_iterator unit_it = unitComboList->begin(); unit_it != unitComboList->end(); ++unit_it ) { + unitBox->insertItem( ( *unit_it ).name ); + unitBox->completionObject() ->addItem( ( *unit_it ).name ); + if ( ( *unit_it ).name != (*unit_it ).plural ) { + unitBox->insertItem( ( *unit_it ).plural ); + unitBox->completionObject() ->addItem( ( *unit_it ).plural ); + } + + if ( !( *unit_it ).name_abbrev.isEmpty() ) { + unitBox->insertItem( ( *unit_it ).name_abbrev ); + unitBox->completionObject() ->addItem( ( *unit_it ).name_abbrev ); + } + if ( !(*unit_it ).plural_abbrev.isEmpty() && + ( *unit_it ).name_abbrev != (*unit_it ).plural_abbrev ) { + unitBox->insertItem( ( *unit_it ).plural_abbrev ); + unitBox->completionObject() ->addItem( ( *unit_it ).plural_abbrev ); + } + + } + } + unitBox->lineEdit() ->setText( store_unit ); +} + +bool IngredientInput::isHeader() const +{ + return typeButtonGrp && (typeButtonGrp->id( typeButtonGrp->selected() ) == 1); +} + +Ingredient IngredientInput::ingredient() const +{ + Ingredient ing; + + ing.prepMethodList = ElementList::split(",",prepMethodBox->currentText()); + ing.name = ingredientBox->currentText(); + amountEdit->value(ing.amount,ing.amount_offset); + ing.units = Unit(unitBox->currentText().stripWhiteSpace(),ing.amount+ing.amount_offset); + ing.ingredientID = ingredientBox->id( ingredientBox->currentItem() ); + + return ing; +} + +TQString IngredientInput::header() const +{ + return headerBox->currentText().stripWhiteSpace(); +} + +void IngredientInput::updateTabOrder() +{ + TQWidget::setTabOrder( ingredientBox, amountEdit ); + TQWidget::setTabOrder( amountEdit, unitBox ); + TQWidget::setTabOrder( unitBox, prepMethodBox ); + TQWidget::setTabOrder( prepMethodBox, orButton ); +} + + +IngredientInputWidget::IngredientInputWidget( RecipeDB *db, TQWidget *parent ) : TQVBox(parent), database(db) +{ + setSizePolicy( TQSizePolicy( TQSizePolicy::Expanding, TQSizePolicy::Fixed ) ); + + m_ingInputs.append(new IngredientInput(database,this)); + + // Connect signals & Slots + connect( m_ingInputs[0], TQ_SIGNAL(addIngredient()), this, TQ_SLOT(addIngredient()) ); + connect( m_ingInputs[0], TQ_SIGNAL(orToggled(bool,IngredientInput*)), this, TQ_SLOT(updateInputs(bool,IngredientInput*)) ); + + reloadCombos(); +} + +IngredientInputWidget::~IngredientInputWidget() +{ +} + +void IngredientInputWidget::clear() +{ + //clearing the first input deletes all substitute inputs + m_ingInputs[0]->clear(); +} + +void IngredientInputWidget::updateInputs(bool on, IngredientInput* input) +{ + TQValueList<IngredientInput*>::iterator curr = m_ingInputs.find(input); + IngredientInput *prev_input = *curr; + ++curr; + + if ( on ) { + IngredientInput *new_input = new IngredientInput(database,this,false); + new_input->reloadCombos(); + + TQWidget::setTabOrder( prev_input, new_input ); + new_input->updateTabOrder(); + + connect( new_input, TQ_SIGNAL(addIngredient()), this, TQ_SLOT(addIngredient()) ); + connect( new_input, TQ_SIGNAL(orToggled(bool,IngredientInput*)), this, TQ_SLOT(updateInputs(bool,IngredientInput*)) ); + + new_input->show(); + m_ingInputs.insert(curr,new_input); + + m_ingInputs[0]->enableHeader(false); + + } + else { + while ( curr != m_ingInputs.end() ) { + (*curr)->deleteLater(); + curr = m_ingInputs.remove(curr); + } + if ( m_ingInputs.count() == 1 ) + m_ingInputs[0]->enableHeader(true); + } +} + +void IngredientInputWidget::addIngredient() +{ + if ( m_ingInputs[0]->isHeader() ) { + TQString header = m_ingInputs[0]->header(); + if ( header.isEmpty() ) + return; + + int group_id = createNewGroupIfNecessary( header,database ); + emit headerEntered( Element(header,group_id) ); + } + else { + for ( TQValueList<IngredientInput*>::iterator it = m_ingInputs.begin(); it != m_ingInputs.end(); ++it ) { + if ( !(*it)->isInputValid() ) + return; + } + + TQValueList<IngredientData> list; + for ( TQValueList<IngredientInput*>::const_iterator it = m_ingInputs.begin(); it != m_ingInputs.end(); ++it ) { + Ingredient ing = (*it)->ingredient(); + ing.ingredientID = createNewIngredientIfNecessary(ing.name,database); + + bool plural = ing.amount+ing.amount_offset > 1; + ing.units.id = createNewUnitIfNecessary( (plural)?ing.units.plural:ing.units.name, plural, ing.ingredientID, ing.units,database ); + if ( ing.units.id == -1 ) // this will happen if the dialog to create a unit was cancelled + return ; + + TQValueList<int> prepIDs = createNewPrepIfNecessary( ing.prepMethodList,database ); + TQValueList<int>::const_iterator id_it = prepIDs.begin(); + for ( ElementList::iterator it = ing.prepMethodList.begin(); it != ing.prepMethodList.end(); ++it, ++id_it ) { + (*it).id = *id_it; + } + + list.append(ing); + } + + Ingredient ing = list.first(); + list.pop_front(); + ing.substitutes = list; + emit ingredientEntered( ing ); + } + clear(); + + m_ingInputs[0]->setFocus(); //put cursor back to the ingredient name so user can begin next ingredient +} + +int IngredientInputWidget::createNewIngredientIfNecessary( const TQString &ing, RecipeDB *database ) +{ + int id = -1; + if ( ing.isEmpty() ) + return -1; + + id = database->findExistingIngredientByName( ing ); + if ( id == -1 ) { + database->createNewIngredient( ing ); + id = database->lastInsertID(); + } + return id; +} + +int IngredientInputWidget::createNewUnitIfNecessary( const TQString &unit, bool plural, int ingredientID, Unit &new_unit, RecipeDB *database ) +{ + int id = database->findExistingUnitByName( unit ); + if ( -1 == id ) + { + CreateUnitDialog getUnit( 0, ( plural ) ? TQString::null : unit, ( !plural ) ? TQString::null : unit ); + if ( getUnit.exec() == TQDialog::Accepted ) { + new_unit = getUnit.newUnit(); + database->createNewUnit( new_unit ); + + id = database->lastInsertID(); + } + } + + if ( !database->ingredientContainsUnit( + ingredientID, + id ) ) + { + database->addUnitToIngredient( + ingredientID, + id ); + } + + new_unit = database->unitName( id ); + + //loadUnitListCombo(); + return id; +} + +TQValueList<int> IngredientInputWidget::createNewPrepIfNecessary( const ElementList &prepMethods, RecipeDB *database ) +{ + TQValueList<int> ids; + + if ( prepMethods.isEmpty() ) //no prep methods + return ids; + else + { + for ( ElementList::const_iterator it = prepMethods.begin(); it != prepMethods.end(); ++it ) { + int id = database->findExistingPrepByName( (*it).name.stripWhiteSpace() ); + if ( id == -1 ) + { + database->createNewPrepMethod( (*it).name.stripWhiteSpace() ); + id = database->lastInsertID(); + } + ids << id; + } + + return ids; + } +} + +int IngredientInputWidget::createNewGroupIfNecessary( const TQString &group, RecipeDB *database ) +{ + if ( group.stripWhiteSpace().isEmpty() ) //no group + return -1; + else + { + int id = database->findExistingIngredientGroupByName( group ); + if ( id == -1 ) //creating new + { + database->createNewIngGroup( group ); + id = database->lastInsertID(); + } + + return id; + } +} + +void IngredientInputWidget::reloadCombos() +{ + for ( TQValueList<IngredientInput*>::iterator it = m_ingInputs.begin(); it != m_ingInputs.end(); ++it ) + (*it)->reloadCombos(); +} + +#include "ingredientinputwidget.moc" diff --git a/src/widgets/ingredientinputwidget.h b/src/widgets/ingredientinputwidget.h new file mode 100644 index 0000000..ecc623a --- /dev/null +++ b/src/widgets/ingredientinputwidget.h @@ -0,0 +1,134 @@ +/*************************************************************************** +* Copyright (C) 2003-2005 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* Jason Kivlighn ([email protected]) * +* * +* Copyright (C) 2006 Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef INGREDIENTINPUTWIDGET_H +#define INGREDIENTINPUTWIDGET_H + +#include <tqvbox.h> + +#include "datablocks/unit.h" + +class TQVBox; +class TQHBox; +class TQLabel; +class TQButtonGroup; +class TQWidgetStack; +class TQGroupBox; +class TQRadioButton; +class TQCheckBox; + +class KComboBox; + +class IngredientComboBox; +class HeaderComboBox; +class PrepMethodComboBox; +class RecipeDB; +class FractionInput; +class Ingredient; +class Element; +class ElementList; +class IngredientInput; + +class IngredientInputWidget : public TQVBox +{ +TQ_OBJECT + +public: + IngredientInputWidget( RecipeDB *db, TQWidget *parent ); + ~IngredientInputWidget(); + + void clear(); + + static int createNewIngredientIfNecessary( const TQString &ing, RecipeDB *db ); + static int createNewUnitIfNecessary( const TQString &unit, bool plural, int ingredient_id, Unit &new_unit, RecipeDB *db ); + static TQValueList<int> createNewPrepIfNecessary( const ElementList &prepMethods, RecipeDB *db ); + static int createNewGroupIfNecessary( const TQString &group, RecipeDB *db ); + +signals: + void ingredientEntered( const Ingredient &ing ); + + void headerEntered( const Element &header ); + +public slots: + void addIngredient(); + +private slots: + void updateInputs(bool,IngredientInput*); + +private: + /** Reloads lists of units, ingredients, and preparation methods */ + void reloadCombos(); + + void checkIfNewUnits(); + + RecipeDB *database; + + TQValueList<IngredientInput*> m_ingInputs; +}; + +class IngredientInput : public TQHBox +{ +TQ_OBJECT + +public: + IngredientInput( RecipeDB *db, TQWidget *parent, bool allowHeader = true ); + ~IngredientInput(); + + void clear(); + bool isInputValid(); + + bool isHeader() const; + Ingredient ingredient() const; + TQString header() const; + + void reloadCombos(); + void enableHeader( bool ); + void updateTabOrder(); + +signals: + void addIngredient(); + void orToggled(bool,IngredientInput*); + +private slots: + void loadUnitListCombo(); + void signalIngredient(); + void typeButtonClicked( int ); + void slotIngredientBoxLostFocus(); + void slotUnitBoxLostFocus(); + void slotPrepMethodBoxLostFocus(); + void orToggled(bool); + +private: + bool checkBounds(); + bool checkAmountEdit(); + + RecipeDB *database; + UnitList *unitComboList; + + TQCheckBox *orButton; + TQGroupBox *ingredientGBox; + TQLabel *amountLabel; + FractionInput* amountEdit; + TQLabel *unitLabel; + KComboBox* unitBox; + TQLabel *prepMethodLabel; + PrepMethodComboBox* prepMethodBox; + TQLabel *ingredientLabel; + IngredientComboBox* ingredientBox; + HeaderComboBox* headerBox; + TQWidgetStack *header_ing_stack; + TQButtonGroup *typeButtonGrp; +}; + +#endif //INGREDIENTINPUTWIDGET_H diff --git a/src/widgets/ingredientlistview.cpp b/src/widgets/ingredientlistview.cpp new file mode 100644 index 0000000..1343530 --- /dev/null +++ b/src/widgets/ingredientlistview.cpp @@ -0,0 +1,287 @@ +/*************************************************************************** +* Copyright (C) 2004 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "ingredientlistview.h" + +#include <tdemessagebox.h> +#include <tdeconfig.h> +#include <tdelocale.h> +#include <tdeglobal.h> +#include <kiconloader.h> +#include <tdepopupmenu.h> + +#include "backends/recipedb.h" +#include "dialogs/createelementdialog.h" +#include "dialogs/dependanciesdialog.h" + +IngredientCheckListItem::IngredientCheckListItem( IngredientCheckListView* qlv, const Element &ing ) : TQCheckListItem( qlv, TQString::null, TQCheckListItem::CheckBox ), + m_listview(qlv) +{ + // Initialize the ingredient data with the the property data + ingStored = new Element(); + ingStored->id = ing.id; + ingStored->name = ing.name; +} + +IngredientCheckListItem::IngredientCheckListItem( IngredientCheckListView* qlv, TQListViewItem *after, const Element &ing ) : TQCheckListItem( qlv, after, TQString::null, TQCheckListItem::CheckBox ), + m_listview(qlv) +{ + // Initialize the ingredient data with the the property data + ingStored = new Element(); + ingStored->id = ing.id; + ingStored->name = ing.name; +} + +IngredientCheckListItem::~IngredientCheckListItem( void ) +{ + delete ingStored; +} +int IngredientCheckListItem::id( void ) const +{ + return ingStored->id; +} +TQString IngredientCheckListItem::name( void ) const +{ + return ingStored->name; +} +Element IngredientCheckListItem::ingredient() const +{ + return *ingStored; +} + +TQString IngredientCheckListItem::text( int column ) const +{ + switch ( column ) { + case 0: + return ( ingStored->name ); + case 1: + return ( TQString::number( ingStored->id ) ); + default: + return TQString::null; + } +} + +void IngredientCheckListItem::stateChange( bool on ) +{ + m_listview->stateChange(this,on); +} + +IngredientListView::IngredientListView( TQWidget *parent, RecipeDB *db ) : DBListViewBase( parent,db, db->ingredientCount() ) +{ + setAllColumnsShowFocus( true ); + setDefaultRenameAction( TQListView::Reject ); +} + +void IngredientListView::init() +{ + connect( database, TQ_SIGNAL( ingredientCreated( const Element & ) ), TQ_SLOT( checkCreateIngredient( const Element & ) ) ); + connect( database, TQ_SIGNAL( ingredientRemoved( int ) ), TQ_SLOT( removeIngredient( int ) ) ); +} + +void IngredientListView::load( int limit, int offset ) +{ + ElementList ingredientList; + database->loadIngredients( &ingredientList, limit, offset ); + + setTotalItems(ingredientList.count()); + + for ( ElementList::const_iterator ing_it = ingredientList.begin(); ing_it != ingredientList.end(); ++ing_it ) + createIngredient( *ing_it ); +} + +void IngredientListView::checkCreateIngredient( const Element &el ) +{ + if ( handleElement(el.name) ) { //only create this ingredient if the base class okays it + createIngredient(el); + } +} + + +StdIngredientListView::StdIngredientListView( TQWidget *parent, RecipeDB *db, bool editable ) : IngredientListView( parent, db ) +{ + addColumn( i18n( "Ingredient" ) ); + + TDEConfig * config = TDEGlobal::config(); + config->setGroup( "Advanced" ); + bool show_id = config->readBoolEntry( "ShowID", false ); + addColumn( i18n( "Id" ), show_id ? -1 : 0 ); + + if ( editable ) { + setRenameable( 0, true ); + + TDEIconLoader *il = new TDEIconLoader; + + kpop = new TDEPopupMenu( this ); + kpop->insertItem( il->loadIcon( "document-new", TDEIcon::NoGroup, 16 ), i18n( "&Create" ), this, TQ_SLOT( createNew() ), CTRL + Key_C ); + kpop->insertItem( il->loadIcon( "edit-delete", TDEIcon::NoGroup, 16 ), i18n( "&Delete" ), this, TQ_SLOT( remove + () ), Key_Delete ); + kpop->insertItem( il->loadIcon( "edit", TDEIcon::NoGroup, 16 ), i18n( "&Rename" ), this, TQ_SLOT( rename() ), CTRL + Key_R ); + kpop->polish(); + + delete il; + + connect( this, TQ_SIGNAL( contextMenu( TDEListView *, TQListViewItem *, const TQPoint & ) ), TQ_SLOT( showPopup( TDEListView *, TQListViewItem *, const TQPoint & ) ) ); + connect( this, TQ_SIGNAL( doubleClicked( TQListViewItem* ) ), this, TQ_SLOT( modIngredient( TQListViewItem* ) ) ); + connect( this, TQ_SIGNAL( itemRenamed( TQListViewItem* ) ), this, TQ_SLOT( saveIngredient( TQListViewItem* ) ) ); + } +} + +void StdIngredientListView::showPopup( TDEListView * /*l*/, TQListViewItem *i, const TQPoint &p ) +{ + if ( i ) + kpop->exec( p ); +} + +void StdIngredientListView::createNew() +{ + CreateElementDialog * elementDialog = new CreateElementDialog( this, i18n( "New Ingredient" ) ); + + if ( elementDialog->exec() == TQDialog::Accepted ) { + TQString result = elementDialog->newElementName(); + + if ( checkBounds( result ) ) + database->createNewIngredient( result ); // Create the new author in the database + } +} + +void StdIngredientListView::remove + () +{ + TQListViewItem * it = currentItem(); + + if ( it ) { + int ingredientID = it->text( 1 ).toInt(); + + ElementList dependingRecipes; + database->findIngredientDependancies( ingredientID, &dependingRecipes ); + + if ( dependingRecipes.isEmpty() ) + database->removeIngredient( ingredientID ); + else { // Need Warning! + ListInfo list; + list.list = dependingRecipes; + list.name = i18n( "Recipes" ); + + DependanciesDialog warnDialog( this, list ); + warnDialog.setCustomWarning( i18n("You are about to permanantly delete recipes from your database.") ); + if ( warnDialog.exec() == TQDialog::Accepted ) + database->removeIngredient( ingredientID ); + } + } +} + +void StdIngredientListView::rename() +{ + TQListViewItem * item = currentItem(); + + if ( item ) + IngredientListView::rename( item, 0 ); +} + +void StdIngredientListView::createIngredient( const Element &ing ) +{ + createElement(new TQListViewItem( this, ing.name, TQString::number( ing.id ) )); +} + +void StdIngredientListView::removeIngredient( int id ) +{ + TQListViewItem * item = findItem( TQString::number( id ), 1 ); + removeElement(item); +} + +void StdIngredientListView::modIngredient( TQListViewItem* i ) +{ + if ( i ) + IngredientListView::rename( i, 0); +} + +void StdIngredientListView::saveIngredient( TQListViewItem* i ) +{ + if ( !checkBounds( i->text( 0 ) ) ) { + reload(ForceReload); //reset the changed text + return ; + } + + int existing_id = database->findExistingIngredientByName( i->text( 0 ) ); + int ing_id = i->text( 1 ).toInt(); + if ( existing_id != -1 && existing_id != ing_id ) //category already exists with this label... merge the two + { + switch ( KMessageBox::warningContinueCancel( this, i18n( "This ingredient already exists. Continuing will merge these two ingredients into one. Are you sure?" ) ) ) + { + case KMessageBox::Continue: { + database->mergeIngredients( existing_id, ing_id ); + break; + } + default: + reload(ForceReload); + break; //we have to reload because the ingredient was renamed, and we need to reset it + } + } + else { + database->modIngredient( ( i->text( 1 ) ).toInt(), i->text( 0 ) ); + } +} + +bool StdIngredientListView::checkBounds( const TQString &name ) +{ + if ( name.length() > uint(database->maxIngredientNameLength()) ) { + KMessageBox::error( this, TQString( i18n( "Ingredient name cannot be longer than %1 characters." ) ).arg( database->maxIngredientNameLength() ) ); + return false; + } + + return true; +} + + + +IngredientCheckListView::IngredientCheckListView( TQWidget *parent, RecipeDB *db ) : IngredientListView( parent, db ) +{ + addColumn( i18n( "Ingredient" ) ); + + TDEConfig *config = TDEGlobal::config(); + config->setGroup( "Advanced" ); + bool show_id = config->readBoolEntry( "ShowID", false ); + addColumn( i18n( "Id" ), show_id ? -1 : 0 ); +} + +void IngredientCheckListView::createIngredient( const Element &ing ) +{ + createElement(new IngredientCheckListItem( this, ing )); +} + +void IngredientCheckListView::removeIngredient( int id ) +{ + TQListViewItem * item = findItem( TQString::number( id ), 1 ); + removeElement(item); +} + +void IngredientCheckListView::load( int limit, int offset ) +{ + IngredientListView::load(limit,offset); + + for ( TQValueList<Element>::const_iterator ing_it = m_selections.begin(); ing_it != m_selections.end(); ++ing_it ) { + TQCheckListItem * item = ( TQCheckListItem* ) findItem( TQString::number( (*ing_it).id ), 1 ); + if ( item ) { + item->setOn(true); + } + } +} + +void IngredientCheckListView::stateChange(IngredientCheckListItem *it,bool on) +{ + if ( !reloading() ) { + if ( on ) + m_selections.append(it->ingredient()); + else + m_selections.remove(it->ingredient()); + } +} + +#include "ingredientlistview.moc" diff --git a/src/widgets/ingredientlistview.h b/src/widgets/ingredientlistview.h new file mode 100644 index 0000000..58990c4 --- /dev/null +++ b/src/widgets/ingredientlistview.h @@ -0,0 +1,119 @@ +/*************************************************************************** +* Copyright (C) 2004 by * +* Jason Kivlighn ([email protected]) * +* Unai Garro ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef INGREDIENTLISTVIEW_H +#define INGREDIENTLISTVIEW_H + +#include "dblistviewbase.h" + +#include "datablocks/element.h" + +class RecipeDB; +class TDEPopupMenu; +class IngredientCheckListView; + +/** +@author Unai Garro +*/ +class IngredientCheckListItem: public TQCheckListItem +{ +public: + IngredientCheckListItem( IngredientCheckListView* qlv, const Element &ing ); + IngredientCheckListItem( IngredientCheckListView* qlv, TQListViewItem *after, const Element &ing ); + ~IngredientCheckListItem( void ); + + int id( void ) const; + TQString name( void ) const; + Element ingredient() const; + + virtual TQString text( int column ) const; + +protected: + virtual void stateChange( bool on ); + +private: + Element *ingStored; + IngredientCheckListView *m_listview; +}; + + + +class IngredientListView : public DBListViewBase +{ + TQ_OBJECT + +public: + IngredientListView( TQWidget *parent, RecipeDB *db ); + +protected slots: + virtual void createIngredient( const Element & ) = 0; + virtual void removeIngredient( int ) = 0; + virtual void load(int limit,int offset); + +protected: + virtual void init(); + +private slots: + virtual void checkCreateIngredient( const Element & ); +}; + + + +class StdIngredientListView : public IngredientListView +{ + TQ_OBJECT + +public: + StdIngredientListView( TQWidget *parent, RecipeDB *db, bool editable = false ); + +protected: + virtual void createIngredient( const Element & ); + virtual void removeIngredient( int ); + +private slots: + void showPopup( TDEListView *, TQListViewItem *, const TQPoint & ); + + void createNew(); + void remove + (); + void rename(); + + void modIngredient( TQListViewItem* i ); + void saveIngredient( TQListViewItem* i ); + +private: + bool checkBounds( const TQString &name ); + + TDEPopupMenu *kpop; +}; + + + +class IngredientCheckListView : public IngredientListView +{ +public: + IngredientCheckListView( TQWidget *parent, RecipeDB *db ); + + virtual void stateChange(IngredientCheckListItem *,bool); + + TQValueList<Element> selections() const{ return m_selections; } + +protected: + virtual void createIngredient( const Element &ing ); + virtual void removeIngredient( int ); + + virtual void load( int limit, int offset ); + +private: + TQValueList<Element> m_selections; +}; + +#endif //INGREDIENTLISTVIEW_H diff --git a/src/widgets/kdateedit.cpp b/src/widgets/kdateedit.cpp new file mode 100644 index 0000000..8becc83 --- /dev/null +++ b/src/widgets/kdateedit.cpp @@ -0,0 +1,388 @@ +/* + This file is part of libtdepim. + + Copyright (c) 2002 Cornelius Schumacher <[email protected]> + Copyright (c) 2003-2004 Reinhold Kainhofer <[email protected]> + Copyright (c) 2004 Tobias Koenig <[email protected]> + + This library is free software; you can redistribute it and/or + modify it under the terms of the GNU Library General Public + License as published by the Free Software Foundation; either + version 2 of the License, or (at your option) any later version. + + This library 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 Library General Public License + along with this library; see the file COPYING.LIB. If not, write to + the Free Software Foundation, Inc., 51 Franklin Steet, Fifth Floor, + Boston, MA 02110-1301 USA +*/ + +#include <tqapplication.h> +#include <tqlineedit.h> +#include <tqlistbox.h> +#include <tqvalidator.h> + +//#include <kcalendarsystem.h> +#include <tdeglobal.h> +#include <tdeglobalsettings.h> +#include <tdelocale.h> +#include <tdeconfig.h> + +#include "kdateedit.h" + +TQRect desktopGeometry(TQWidget* w) +{ + TQDesktopWidget *dw = TQApplication::desktop(); + if (dw->isVirtualDesktop()) { + TDEConfigGroup group(TDEGlobal::config(), "Windows"); + if (group.readBoolEntry("XineramaEnabled", true) && + group.readBoolEntry("XineramaPlacementEnabled", true)) { + if (w) + return dw->screenGeometry(dw->screenNumber(w)); + else return dw->screenGeometry(-1); + } else { + return dw->geometry(); + } + } else { + return dw->geometry(); + } +} + +class DateValidator : public TQValidator +{ + public: + DateValidator( const TQStringList &keywords, TQWidget* parent, const char* name = 0 ) + : TQValidator( parent, name ), mKeywords( keywords ) + {} + + virtual State validate( TQString &str, int& ) const + { + int length = str.length(); + + // empty string is intermediate so one can clear the edit line and start from scratch + if ( length <= 0 ) + return Intermediate; + + if ( mKeywords.contains( str.lower() ) ) + return Acceptable; + + bool ok = false; + TDEGlobal::locale()->readDate( str, &ok ); + if ( ok ) + return Acceptable; + else + return Intermediate; + } + + private: + TQStringList mKeywords; +}; + +KDateEdit::KDateEdit( TQWidget *parent, const char *name ) + : TQComboBox( true, parent, name ), + mReadOnly( false ), + mDiscardNextMousePress( false ) +{ + // need at least one entry for popup to work + setMaxCount( 1 ); + + mDate = TQDate::currentDate(); + TQString today = TDEGlobal::locale()->formatDate( mDate, true ); + + insertItem( today ); + setCurrentItem( 0 ); + changeItem( today, 0 ); + setMinimumSize( sizeHint() ); + + connect( lineEdit(), TQ_SIGNAL( returnPressed() ), + this, TQ_SLOT( lineEnterPressed() ) ); + connect( this, TQ_SIGNAL( textChanged( const TQString& ) ), + TQ_SLOT( slotTextChanged( const TQString& ) ) ); + + mPopup = new KDatePickerPopup( KDatePickerPopup::DatePicker | KDatePickerPopup::Words ); + mPopup->hide(); + mPopup->installEventFilter( this ); + + connect( mPopup, TQ_SIGNAL( dateChanged( TQDate ) ), + TQ_SLOT( dateSelected( TQDate ) ) ); + + // handle keyword entry + setupKeywords(); + lineEdit()->installEventFilter( this ); + + setValidator( new DateValidator( mKeywordMap.keys(), this ) ); + + mTextChanged = false; +} + +KDateEdit::~KDateEdit() +{ + delete mPopup; + mPopup = 0; +} + +void KDateEdit::setDate( const TQDate& date ) +{ + assignDate( date ); + updateView(); +} + +TQDate KDateEdit::date() const +{ + return mDate; +} + +void KDateEdit::setReadOnly( bool readOnly ) +{ + mReadOnly = readOnly; + lineEdit()->setReadOnly( readOnly ); +} + +bool KDateEdit::isReadOnly() const +{ + return mReadOnly; +} + +void KDateEdit::popup() +{ + if ( mReadOnly ) + return; + + TQRect desk = desktopGeometry( this ); + + TQPoint popupPoint = mapToGlobal( TQPoint( 0,0 ) ); + + int dateFrameHeight = mPopup->sizeHint().height(); + if ( popupPoint.y() + height() + dateFrameHeight > desk.bottom() ) + popupPoint.setY( popupPoint.y() - dateFrameHeight ); + else + popupPoint.setY( popupPoint.y() + height() ); + + int dateFrameWidth = mPopup->sizeHint().width(); + if ( popupPoint.x() + dateFrameWidth > desk.right() ) + popupPoint.setX( desk.right() - dateFrameWidth ); + + if ( popupPoint.x() < desk.left() ) + popupPoint.setX( desk.left() ); + + if ( popupPoint.y() < desk.top() ) + popupPoint.setY( desk.top() ); + + if ( mDate.isValid() ) + mPopup->setDate( mDate ); + else + mPopup->setDate( TQDate::currentDate() ); + + mPopup->popup( popupPoint ); + + // The combo box is now shown pressed. Make it show not pressed again + // by causing its (invisible) list box to emit a 'selected' signal. + // First, ensure that the list box contains the date currently displayed. + TQDate date = parseDate(); + assignDate( date ); + updateView(); + // Now, simulate an Enter to unpress it + TQListBox *lb = listBox(); + if (lb) { + lb->setCurrentItem(0); + TQKeyEvent* keyEvent = new TQKeyEvent(TQEvent::KeyPress, TQt::Key_Enter, 0, 0); + TQApplication::postEvent(lb, keyEvent); + } +} + +void KDateEdit::dateSelected( TQDate date ) +{ + if (assignDate( date ) ) { + updateView(); + emit dateChanged( date ); + + if ( date.isValid() ) { + mPopup->hide(); + } + } +} + +void KDateEdit::dateEntered( TQDate date ) +{ + if (assignDate( date ) ) { + updateView(); + emit dateChanged( date ); + } +} + +void KDateEdit::lineEnterPressed() +{ + bool replaced = false; + + TQDate date = parseDate( &replaced ); + + if (assignDate( date ) ) { + if ( replaced ) + updateView(); + + emit dateChanged( date ); + } +} + +TQDate KDateEdit::parseDate( bool *replaced ) const +{ + TQString text = currentText(); + TQDate result; + + if ( replaced ) + (*replaced) = false; + + if ( text.isEmpty() ) + result = TQDate(); + else if ( mKeywordMap.contains( text.lower() ) ) { + TQDate today = TQDate::currentDate(); + int i = mKeywordMap[ text.lower() ]; + if ( i >= 100 ) { + /* A day name has been entered. Convert to offset from today. + * This uses some math tricks to figure out the offset in days + * to the next date the given day of the week occurs. There + * are two cases, that the new day is >= the current day, which means + * the new day has not occurred yet or that the new day < the current day, + * which means the new day is already passed (so we need to find the + * day in the next week). + */ + i -= 100; + int currentDay = today.dayOfWeek(); + if ( i >= currentDay ) + i -= currentDay; + else + i += 7 - currentDay; + } + + result = today.addDays( i ); + if ( replaced ) + (*replaced) = true; + } else { + result = TDEGlobal::locale()->readDate( text ); + } + + return result; +} + +bool KDateEdit::eventFilter( TQObject *object, TQEvent *event ) +{ + if ( object == lineEdit() ) { + // We only process the focus out event if the text has changed + // since we got focus + if ( (event->type() == TQEvent::FocusOut) && mTextChanged ) { + lineEnterPressed(); + mTextChanged = false; + } else if ( event->type() == TQEvent::KeyPress ) { + // Up and down arrow keys step the date + TQKeyEvent* keyEvent = (TQKeyEvent*)event; + + if ( keyEvent->key() == TQt::Key_Return ) { + lineEnterPressed(); + return true; + } + + int step = 0; + if ( keyEvent->key() == TQt::Key_Up ) + step = 1; + else if ( keyEvent->key() == TQt::Key_Down ) + step = -1; + if ( step && !mReadOnly ) { + TQDate date = parseDate(); + if ( date.isValid() ) { + date = date.addDays( step ); + if ( assignDate( date ) ) { + updateView(); + emit dateChanged( date ); + return true; + } + } + } + } + } else { + // It's a date picker event + switch ( event->type() ) { + case TQEvent::MouseButtonDblClick: + case TQEvent::MouseButtonPress: { + TQMouseEvent *mouseEvent = (TQMouseEvent*)event; + if ( !mPopup->rect().contains( mouseEvent->pos() ) ) { + TQPoint globalPos = mPopup->mapToGlobal( mouseEvent->pos() ); + if ( TQApplication::widgetAt( globalPos, true ) == this ) { + // The date picker is being closed by a click on the + // KDateEdit widget. Avoid popping it up again immediately. + mDiscardNextMousePress = true; + } + } + + break; + } + default: + break; + } + } + + return false; +} + +void KDateEdit::mousePressEvent( TQMouseEvent *event ) +{ + if ( event->button() == TQt::LeftButton && mDiscardNextMousePress ) { + mDiscardNextMousePress = false; + return; + } + + TQComboBox::mousePressEvent( event ); +} + +void KDateEdit::slotTextChanged( const TQString& ) +{ + TQDate date = parseDate(); + + if ( assignDate( date ) ) + emit dateChanged( date ); + + mTextChanged = true; +} + +void KDateEdit::setupKeywords() +{ + // Create the keyword list. This will be used to match against when the user + // enters information. + mKeywordMap.insert( i18n( "tomorrow" ), 1 ); + mKeywordMap.insert( i18n( "today" ), 0 ); + mKeywordMap.insert( i18n( "yesterday" ), -1 ); + + #if 0 //depends on KDE 3.2 + TQString dayName; + for ( int i = 1; i <= 7; ++i ) { + dayName = TDEGlobal::locale()->calendar()->weekDayName( i ).lower(); + mKeywordMap.insert( dayName, i + 100 ); + } + #endif +} + +bool KDateEdit::assignDate( const TQDate& date ) +{ + mDate = date; + mTextChanged = false; + return true; +} + +void KDateEdit::updateView() +{ + TQString dateString; + if ( mDate.isValid() ) + dateString = TDEGlobal::locale()->formatDate( mDate, true ); + + // We do not want to generate a signal here, + // since we explicitly setting the date + bool blocked = signalsBlocked(); + blockSignals( true ); + changeItem( dateString, 0 ); + blockSignals( blocked ); +} + +#include "kdateedit.moc" diff --git a/src/widgets/kdateedit.h b/src/widgets/kdateedit.h new file mode 100644 index 0000000..e3957b2 --- /dev/null +++ b/src/widgets/kdateedit.h @@ -0,0 +1,139 @@ +/* + This file is part of libtdepim. + + Copyright (c) 2002 Cornelius Schumacher <[email protected]> + Copyright (c) 2004 Tobias Koenig <[email protected]> + + This library is free software; you can redistribute it and/or + modify it under the terms of the GNU Library General Public + License as published by the Free Software Foundation; either + version 2 of the License, or (at your option) any later version. + + This library 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 Library General Public License + along with this library; see the file COPYING.LIB. If not, write to + the Free Software Foundation, Inc., 51 Franklin Steet, Fifth Floor, + Boston, MA 02110-1301 USA +*/ + +#ifndef KDATEEDIT_H +#define KDATEEDIT_H + +#include <tqcombobox.h> +#include <tqdatetime.h> +#include <tqmap.h> + +#include "kdatepickerpopup.h" + +class TQEvent; + +/** + A date editing widget that consists of an editable combo box. + The combo box contains the date in text form, and clicking the combo + box arrow will display a 'popup' style date picker. + + This widget also supports advanced features like allowing the user + to type in the day name to get the date. The following keywords + are supported (in the native language): tomorrow, yesturday, today, + monday, tuesday, wednesday, thursday, friday, saturday, sunday. + + @image html kdateedit.png "This is how it looks" + + @author Cornelius Schumacher <[email protected]> + @author Mike Pilone <[email protected]> + @author David Jarvie <[email protected]> + @author Tobias Koenig <[email protected]> +*/ +class KDateEdit : public TQComboBox +{ + TQ_OBJECT + + public: + KDateEdit( TQWidget *parent = 0, const char *name = 0 ); + virtual ~KDateEdit(); + + /** + @return The date entered. This date could be invalid, + you have to check validity yourself. + */ + TQDate date() const; + + /** + Sets whether the widget is read-only for the user. If read-only, + the date picker pop-up is inactive, and the displayed date cannot be edited. + + @param readOnly True to set the widget read-only, false to set it read-write. + */ + void setReadOnly( bool readOnly ); + + /** + @return True if the widget is read-only, false if read-write. + */ + bool isReadOnly() const; + + virtual void popup(); + + signals: + /** + This signal is emitted whenever the user modifies the date. + The passed date can be invalid. + */ + void dateChanged( const TQDate &date ); + + public slots: + /** + Sets the date. + + @param date The new date to display. This date must be valid or + it will not be set + */ + void setDate( const TQDate &date ); + + protected slots: + void lineEnterPressed(); + void slotTextChanged( const TQString& ); + void dateEntered( TQDate ); + void dateSelected( TQDate ); + + protected: + virtual bool eventFilter( TQObject*, TQEvent* ); + virtual void mousePressEvent( TQMouseEvent* ); + + /** + Sets the date, without altering the display. + This method is used internally to set the widget's date value. + As a virtual method, it allows derived classes to perform additional validation + on the date value before it is set. Derived classes should return true if + TQDate::isValid(@p date) returns false. + + @param date The new date to set. + @return True if the date was set, false if it was considered invalid and + remains unchanged. + */ + virtual bool assignDate( const TQDate &date ); + + /** + Fills the keyword map. Reimplement it if you want additional + keywords. + */ + void setupKeywords(); + + private: + TQDate parseDate( bool* = 0 ) const; + void updateView(); + + KDatePickerPopup *mPopup; + + TQDate mDate; + bool mReadOnly; + bool mTextChanged; + bool mDiscardNextMousePress; + + TQMap<TQString, int> mKeywordMap; +}; + +#endif diff --git a/src/widgets/kdatepickerpopup.cpp b/src/widgets/kdatepickerpopup.cpp new file mode 100644 index 0000000..da68918 --- /dev/null +++ b/src/widgets/kdatepickerpopup.cpp @@ -0,0 +1,123 @@ +/* + This file is part of libtdepim. + + Copyright (c) 2004 Bram Schoenmakers <[email protected]> + + This library is free software; you can redistribute it and/or + modify it under the terms of the GNU Library General Public + License as published by the Free Software Foundation; either + version 2 of the License, or (at your option) any later version. + + This library 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 Library General Public License + along with this library; see the file COPYING.LIB. If not, write to + the Free Software Foundation, Inc., 51 Franklin Steet, Fifth Floor, + Boston, MA 02110-1301 USA +*/ + +#include <tqdatetime.h> +#include <tqpopupmenu.h> + +#include <tdelocale.h> + +#include "kdatepickerpopup.h" + +KDatePickerPopup::KDatePickerPopup( int items, const TQDate &date, TQWidget *parent, + const char *name ) + : TQPopupMenu( parent, name ) +{ + mItems = items; + + mDatePicker = new KDatePicker( this ); + mDatePicker->setCloseButton( false ); + + connect( mDatePicker, TQ_SIGNAL( dateEntered( TQDate ) ), + TQ_SLOT( slotDateChanged( TQDate ) ) ); + connect( mDatePicker, TQ_SIGNAL( dateSelected( TQDate ) ), + TQ_SLOT( slotDateChanged( TQDate ) ) ); + + mDatePicker->setDate( date ); + + buildMenu(); +} + +void KDatePickerPopup::buildMenu() +{ + if ( isVisible() ) return; + clear(); + + if ( mItems & DatePicker ) { + insertItem( mDatePicker ); + + if ( ( mItems & NoDate ) || ( mItems & Words ) ) + insertSeparator(); + } + + if ( mItems & Words ) { + insertItem( i18n("&Today"), this, TQ_SLOT( slotToday() ) ); + insertItem( i18n("&Yesterday"), this, TQ_SLOT( slotYesterday() ) ); + insertItem( i18n("Last &Week"), this, TQ_SLOT( slotLastWeek() ) ); + insertItem( i18n("Last M&onth"), this, TQ_SLOT( slotLastMonth() ) ); + + if ( mItems & NoDate ) + insertSeparator(); + } + + if ( mItems & NoDate ) + insertItem( i18n("No Date"), this, TQ_SLOT( slotNoDate() ) ); +} + +KDatePicker *KDatePickerPopup::datePicker() const +{ + return mDatePicker; +} + +void KDatePickerPopup::setDate( const TQDate &date ) +{ + mDatePicker->setDate( date ); +} + +#if 0 +void KDatePickerPopup::setItems( int items ) +{ + mItems = items; + buildMenu(); +} +#endif + +void KDatePickerPopup::slotDateChanged( TQDate date ) +{ + emit dateChanged( date ); + hide(); +} + +void KDatePickerPopup::slotToday() +{ + emit dateChanged( TQDate::currentDate() ); +} + +void KDatePickerPopup::slotYesterday() +{ + emit dateChanged( TQDate::currentDate().addDays( -1 ) ); +} + +void KDatePickerPopup::slotNoDate() +{ + emit dateChanged( TQDate() ); +} + +void KDatePickerPopup::slotLastWeek() +{ + emit dateChanged( TQDate::currentDate().addDays( -7 ) ); +} + +void KDatePickerPopup::slotLastMonth() +{ + emit dateChanged( TQDate::currentDate().addMonths( -1 ) ); +} + +#include "kdatepickerpopup.moc" diff --git a/src/widgets/kdatepickerpopup.h b/src/widgets/kdatepickerpopup.h new file mode 100644 index 0000000..67f36dd --- /dev/null +++ b/src/widgets/kdatepickerpopup.h @@ -0,0 +1,102 @@ +/* + This file is part of libtdepim. + + Copyright (c) 2004 Bram Schoenmakers <[email protected]> + + This library is free software; you can redistribute it and/or + modify it under the terms of the GNU Library General Public + License as published by the Free Software Foundation; either + version 2 of the License, or (at your option) any later version. + + This library 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 Library General Public License + along with this library; see the file COPYING.LIB. If not, write to + the Free Software Foundation, Inc., 51 Franklin Steet, Fifth Floor, + Boston, MA 02110-1301 USA +*/ +#ifndef KDATEPICKERPOPUP_H +#define KDATEPICKERPOPUP_H + +#include <tqdatetime.h> +#include <tqpopupmenu.h> + +#include <kdatepicker.h> + +/** + @short This menu helps the user to select a date quickly. + + This menu helps the user to select a date tquicly. It offers various ways of selecting, e.g. with a KDatePicker or with words like "Tomorrow". + + The available items are: + + @li NoDate: A menu-item with "No Date". If choosen, the datepicker will emit a null TQDate. + @li DatePicker: Show a KDatePicker-widget. + @li Words: Show items like "Today", "Tomorrow" or "Next Week". + + When supplying multiple items, separate each item with a bitwise OR. + + @author Bram Schoenmakers <[email protected]> +*/ +class KDatePickerPopup: public TQPopupMenu +{ + TQ_OBJECT + public: + enum { NoDate = 1, DatePicker = 2, Words = 4 }; + + /** + A constructor for the KDatePickerPopup. + + @param items List of all desirable items, separated with a bitwise OR. + @param date Initial date of datepicker-widget. + @param parent The object's parent. + @param name The object's name. + */ + KDatePickerPopup( int items = 2, const TQDate &date = TQDate::currentDate(), + TQWidget *parent = 0, const char *name = 0 ); + + /** + @return A pointer to the private variable mDatePicker, an instance of + KDatePicker. + */ + KDatePicker *datePicker() const; + + void setDate( const TQDate &date ); + +#if 0 + /** Set items which should be shown and rebuilds the menu afterwards. Only if the menu is not visible. + @param items List of all desirable items, separated with a bitwise OR. + */ + void setItems( int items = 1 ); +#endif + /** @return Returns the bitwise result of the active items in the popup. */ + int items() const { return mItems; } + + signals: + + /** + This signal emits the new date (selected with datepicker or other + menu-items). + */ + void dateChanged ( TQDate ); + + protected slots: + void slotDateChanged ( TQDate ); + + void slotToday(); + void slotYesterday(); + void slotLastWeek(); + void slotLastMonth(); + void slotNoDate(); + + private: + void buildMenu(); + + KDatePicker *mDatePicker; + int mItems; +}; + +#endif diff --git a/src/widgets/krelistview.cpp b/src/widgets/krelistview.cpp new file mode 100644 index 0000000..3c06216 --- /dev/null +++ b/src/widgets/krelistview.cpp @@ -0,0 +1,116 @@ +/*************************************************************************** +* Copyright (C) 2004-2005 by * +* Jason Kivlighn ([email protected]) * +* * +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "krelistview.h" + +#include <tdeglobalsettings.h> +#include <tdelocale.h> +#include <kdebug.h> + +#include "widgets/dblistviewbase.h" + +KreListView::KreListView( TQWidget *parent, const TQString &title, bool filter, int filterCol, TQWidget *embeddedWidget ) : TQVBox( parent ) +{ + + filteredColumn = filterCol; + TQWidget *header = this; + if ( filter || embeddedWidget ) { + header = new TQHBox( this ); + ( ( TQHBox* ) header ) ->setSpacing( 15 ); + } + + if ( !title.isNull() ) { + listLabel = new TQLabel( header ); + listLabel->setFrameShape( TQFrame::GroupBoxPanel ); + listLabel->setFrameShadow( TQFrame::Sunken ); + listLabel->setPaletteForegroundColor( TDEGlobalSettings::highlightedTextColor() ); + listLabel->setPaletteBackgroundColor( TDEGlobalSettings::highlightColor().light( 120 ) ); // 120, to match the kremenu settings + listLabel->setText( title ); + + } + + if ( filter ) { + filterBox = new TQHBox( header ); + filterBox->setFrameShape( TQFrame::Box ); + filterBox->setMargin( 2 ); + filterLabel = new TQLabel( filterBox ); + filterLabel->setText( i18n( "Search:" ) ); + filterEdit = new KLineEdit( filterBox ); + } + + + list = new TDEListView( this ); + list->setSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::MinimumExpanding ); + setSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::MinimumExpanding ); + setSpacing( 10 ); + + + // If the user provides a widget, embed it into the header + if ( embeddedWidget ) + embeddedWidget->reparent( header, TQPoint( 0, 0 ) ); + //Connect Signals & Slots + if ( filter ) { + connect( filterEdit, TQ_SIGNAL( textChanged( const TQString& ) ), TQ_SIGNAL( textChanged(const TQString&) ) ); + connect( this, TQ_SIGNAL( textChanged( const TQString& ) ), TQ_SLOT( filter( const TQString& ) ) ); + } +} + +KreListView::~KreListView() +{} + +void KreListView::filter( const TQString& s ) +{ + for ( TQListViewItem * it = list->firstChild();it;it = it->nextSibling() ) { + if ( it->rtti() == NEXTLISTITEM_RTTI || it->rtti() == PREVLISTITEM_RTTI ) + continue; + + if ( s.isEmpty() ) // Don't filter if the filter text is empty + { + it->setVisible( true ); + } + else + { + + if ( it->text( filteredColumn ).contains( s, false ) ) + it->setVisible( true ); + else + it->setVisible( false ); + + } + + + } +} + +void KreListView::refilter() +{ + if ( !filterEdit->text().isEmpty() ) { + emit textChanged( filterEdit->text() ); + } +} + +void KreListView::setCustomFilter( TQObject *receiver, const char *slot ) +{ + connect( this, TQ_SIGNAL( textChanged( const TQString& ) ), receiver, slot ); +} + +void KreListView::setListView( DBListViewBase *list_view ) +{ + delete list; + + connect( list_view, TQ_SIGNAL( nextGroupLoaded() ), TQ_SLOT( refilter() ) ); + connect( list_view, TQ_SIGNAL( prevGroupLoaded() ), TQ_SLOT( refilter() ) ); + list = list_view; +} + +#include "krelistview.moc" diff --git a/src/widgets/krelistview.h b/src/widgets/krelistview.h new file mode 100644 index 0000000..0884bda --- /dev/null +++ b/src/widgets/krelistview.h @@ -0,0 +1,65 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef KRELISTVIEW_H +#define KRELISTVIEW_H + +#include <tqlabel.h> +#include <tqvbox.h> +#include <tdelistview.h> +#include <klineedit.h> + +class DBListViewBase; + +/** +@author Unai Garro +*/ + +class KreListView: public TQVBox +{ + TQ_OBJECT +public: + + KreListView( TQWidget *parent, const TQString &title = TQString::null, bool filter = false, int filterCol = 0, TQWidget *embeddedWidget = 0 ); + ~KreListView(); + TDEListView *listView() + { + return list; + } + + void setListView( TDEListView *list_view ) + { + delete list; + list = list_view; + } + void setListView( DBListViewBase *list_view ); + + void setCustomFilter( TQObject *receiver, const char *slot ); + TQString filterText() const { return filterEdit->text(); } + +public slots: + void refilter(); + +signals: + void textChanged( const TQString & ); + +private: + TQHBox *filterBox; + TQLabel *listLabel; + int filteredColumn; + TQLabel *filterLabel; + KLineEdit *filterEdit; + TDEListView *list; + +private slots: + void filter( const TQString& s ); +}; + +#endif diff --git a/src/widgets/kremenu.cpp b/src/widgets/kremenu.cpp new file mode 100644 index 0000000..365fba9 --- /dev/null +++ b/src/widgets/kremenu.cpp @@ -0,0 +1,631 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* * +* * +* This program is free software; you can redistribute it and/or modify * +* it under the terms of the GNU General Public License as publishfed by * +* the Free Software Foundation; either version 2 of the License, or * +* (at your option) any later version. * +***************************************************************************/ +#include "kremenu.h" + +#include <tqbitmap.h> +#include <tqcursor.h> +#include <tqfont.h> +#include <tqimage.h> +#include <tqobjectlist.h> +#include <tqpainter.h> +#include <tqpixmap.h> +#include <tqsignalmapper.h> + +#include <tdeapplication.h> +#include <kcursor.h> +#include <kdebug.h> +#include <tdeglobalsettings.h> +#include <kiconloader.h> +#include <kimageeffect.h> +#include <tdelocale.h> +#include <kpixmap.h> +#include <kpixmapeffect.h> + +KreMenu::KreMenu( TQWidget *parent, const char *name ) : + TQWidget( parent, name, TQt::WNoAutoErase ) +{ + Menu newMenu; + + mainMenuId = menus.append( newMenu ); + + currentMenuId = mainMenuId; + m_currentMenu = &( *currentMenuId ); + + setMouseTracking( true ); + setFocusPolicy( TQWidget::StrongFocus ); + setSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Preferred ); +} + + +KreMenu::~KreMenu() +{} + +void KreMenu::childEvent ( TQChildEvent *e ) +{ + if ( e->type() == TQChildEvent::ChildInserted ) { + + TQObject * child = e->child(); + if ( child->inherits( "KreMenuButton" ) ) { + KreMenuButton * button = ( KreMenuButton* ) child; + + Menu *buttonMenu = &( *( button->menuId ) ); + + + + if ( !( buttonMenu->activeButton ) ) // Highlight the button if it's the first in the menu + { + button->setActive( true ); + buttonMenu->activeButton = button; + } + + buttonMenu->addButton( button ); + + if ( buttonMenu != m_currentMenu ) + button->hide(); + else + button->show(); + + connect ( button, TQ_SIGNAL( clicked( KreMenuButton* ) ), this, TQ_SLOT( collectClicks( KreMenuButton* ) ) ); + } + } + else if ( e->type() == TQChildEvent::ChildRemoved ) { + TQObject * child = e->child(); + + KreMenuButton *button = ( KreMenuButton* ) child; + if ( m_currentMenu->positionList.find( button ) != m_currentMenu->positionList.end() ) // Ensure that what was removed was a button + { + // Remove the button from the list first + int pos = m_currentMenu->positionList[ button ]; // FIXME: this works only if the button is removed from the main menu + m_currentMenu->widgetList.remove( pos ); // FIXME: this works only if the button is removed from the main menu + m_currentMenu->positionList.remove( button ); // FIXME: this works only if the button is removed from the main menu + + // Now recalculate the position of the next button + ( m_currentMenu->widgetNumber ) --; // FIXME: this works only if the button is removed from the main menu + + KreMenuButton *lastButton = m_currentMenu->widgetList[ ( m_currentMenu->widgetNumber ) - 1 ]; + if ( lastButton ) + m_currentMenu->childPos = lastButton->y() + lastButton->height(); + m_currentMenu->activeButton = 0; + + setMinimumWidth( minimumSizeHint().width() + 10 ); //update the minimum width + } + + } + TQWidget::childEvent( e ); +} + +void KreMenu::collectClicks( KreMenuButton *w ) +{ + setFocus(); + + highlightButton( w ); + + // Emit signal indicating button activation with button ID + KrePanel panel = w->getPanel(); + emit clicked( panel ); +} + +MenuId KreMenu::createSubMenu( const TQString &title, const TQString &icon ) +{ + + // Create the new menu + Menu newMenu; + MenuId id = menus.append( newMenu ); + + // Add a button to the main menu for this submenu + TDEIconLoader il; + KreMenuButton *newMenuButton = new KreMenuButton( this ); + newMenuButton->subMenuId = id; + newMenuButton->setTitle( title ); + newMenuButton->setIconSet( il.loadIconSet( icon, TDEIcon::Panel ) ); + + // Add a button to the submenu to go back to the top menu + KreMenuButton *newSubMenuButton = new KreMenuButton( this ); + newSubMenuButton->menuId = id; + newSubMenuButton->subMenuId = mainMenuId; + newSubMenuButton->setTitle( i18n( "Up" ) ); + newSubMenuButton->setIconSet( il.loadIconSet( "1uparrow", TDEIcon::Panel ) ); + + connect( newMenuButton, TQ_SIGNAL( clicked( MenuId ) ), this, TQ_SLOT( showMenu( MenuId ) ) ); + connect( newSubMenuButton, TQ_SIGNAL( clicked( MenuId ) ), this, TQ_SLOT( showMenu( MenuId ) ) ); + + + return id; +} + +void KreMenu::highlightButton( KreMenuButton *button ) +{ + MenuId buttonMenuId = button->menuId; + Menu *buttonMenu = &( *buttonMenuId ); + + //Deactivate the old button + if ( buttonMenu->activeButton ) { + buttonMenu->activeButton->setActive( false ); + buttonMenu->activeButton->update(); + } + + //Activate the new button + + button->setActive( true ); + button->update(); + buttonMenu->activeButton = button; +} + +void KreMenu::keyPressEvent( TQKeyEvent *e ) +{ + switch ( e->key() ) { + case TQt::Key_Up: { + int current_index = m_currentMenu->positionList[ m_currentMenu->activeButton ]; + if ( current_index > 0 ) { + highlightButton( m_currentMenu->widgetList[ current_index - 1 ] ); + + //simulate a mouse click + TQMouseEvent me( TQEvent::MouseButtonPress, TQPoint(), 0, 0 ); + TDEApplication::sendEvent( m_currentMenu->activeButton, &me ); + + e->accept(); + } + break; + } + case TQt::Key_Down: { + int current_index = m_currentMenu->positionList[ m_currentMenu->activeButton ]; + if ( current_index < int( m_currentMenu->positionList.count() ) - 1 ) { + highlightButton( m_currentMenu->widgetList[ current_index + 1 ] ); + + //simulate a mouse click + TQMouseEvent me( TQEvent::MouseButtonPress, TQPoint(), 0, 0 ); + TDEApplication::sendEvent( m_currentMenu->activeButton, &me ); + + e->accept(); + } + break; + } + case TQt::Key_Enter: + case TQt::Key_Return: + case TQt::Key_Space: { + //simulate a mouse click + TQMouseEvent me( TQEvent::MouseButtonPress, TQPoint(), 0, 0 ); + TDEApplication::sendEvent( m_currentMenu->activeButton, &me ); + + e->accept(); + break; + } + default: + e->ignore(); + } +} + +TQSize KreMenu::sizeHint() const +{ + return minimumSizeHint(); +} + +//the minimum size hint will be the minimum size hint of the largest child +TQSize KreMenu::minimumSizeHint() const +{ + int width = 30; + + TQObjectList *childElements = queryList( 0, 0, false, false ); //only first-generation children (not recursive) + TQObjectListIterator it( *childElements ); + + TQObject *obj; + while ( ( obj = it.current() ) != 0 ) { + ++it; + + if ( obj->isWidgetType() ) { + int obj_width_hint = ( ( TQWidget* ) obj ) ->minimumSizeHint().width(); + + if ( obj_width_hint > width ) + width = obj_width_hint; + } + } + + return TQSize( width, 150 ); +} + +void KreMenu::paintEvent( TQPaintEvent * ) +{ + // Make sure the size is bigger than the minimum necessary + //if (minimumWidth() <45) setMinimumWidth(45); // FIXME: can somehow setMinimumWidth be restricted? This may not be the best place to do this + + // Get gradient colors + TQColor c = colorGroup().button(); + TQColor c1 = c.dark( 130 ); + TQColor c2 = c.light( 120 ); + + // Draw the gradient + KPixmap kpm; + kpm.resize( size() ); + KPixmapEffect::unbalancedGradient ( kpm, c2, c1, KPixmapEffect::HorizontalGradient, -150, -150 ); + + // Draw the handle + TQPainter painter( &kpm ); + painter.setPen( c1 ); + painter.drawLine( width() - 5, 20, width() - 5, height() - 20 ); + painter.end(); + + //Set the border transparent using a mask + TQBitmap mask( kpm.size() ); + mask.fill( TQt::color0 ); + painter.begin( &mask ); + painter.setPen( TQt::color1 ); + painter.setBrush( TQt::color1 ); + painter.drawRoundRect( 0, 0, width(), height(), ( int ) ( 2.0 / width() * height() ), 2 ); + painter.end(); + kpm.setMask( mask ); + + //Draw the border line + painter.begin( &kpm ); + painter.setPen( c1 ); + painter.drawRoundRect( 0, 0, width(), height(), ( int ) ( 2.0 / width() * height() ), 2 ); + + //Draw the top line bordering with the first button + if ( m_currentMenu->activeButton ) // draw only if there's a button + { + int w = m_currentMenu->activeButton->width(); + painter.setPen( c1 ); + painter.drawLine( w / 5, 8, w - 1, 8 ); + painter.setPen( c2 ); + painter.drawLine( w / 5, 9, w - 1, 9 ); + } + + painter.end(); + + // Copy the pixmap to the widget + bitBlt( this, 0, 0, &kpm ); +} + +void KreMenu::resizeEvent( TQResizeEvent* e ) +{ + emit resized( ( e->size() ).width(), ( e->size() ).height() ); +} + +void KreMenu::showMenu( MenuId id ) +{ + + // Hide the buttons in the current menu + // and show the ones in the new menu + + TQObjectList * childElements = queryList(); + TQObjectListIterator it( *childElements ); + + TQObject *obj; + while ( ( obj = it.current() ) != 0 ) { + ++it; + if ( obj->inherits( "KreMenuButton" ) ) { + KreMenuButton * button = ( KreMenuButton* ) obj; + if ( button->menuId == currentMenuId ) + button->hide(); + else if ( button->menuId == id ) + button->show(); + } + } + + + // Set the new menu as current + currentMenuId = id; + m_currentMenu = &( *( currentMenuId ) ); +} + + + + + +KreMenuButton::KreMenuButton( KreMenu *parent, KrePanel _panel, MenuId id, const char *name ) : + TQWidget( parent, name, TQt::WNoAutoErase ), + panel( _panel ) +{ + icon = 0; + highlighted = false; + text = TQString::null; + + if ( id == 0 ) + menuId = parent->mainMenu(); + else + menuId = id; + + subMenuId = 0; // By default it's not a submenu button + + resize( parent->size().width(), 55 ); + connect ( parent, TQ_SIGNAL( resized( int, int ) ), this, TQ_SLOT( rescale() ) ); + connect( this, TQ_SIGNAL( clicked() ), this, TQ_SLOT( forwardClicks() ) ); + setCursor( TQCursor( KCursor::handCursor() ) ); +} + + +KreMenuButton::~KreMenuButton() +{ + delete icon; +} + +void KreMenuButton::setTitle( const TQString &s ) +{ + text = s; + +#if 0 //this causes problems for the button to go back to editing a recipe + //adjust text to two lines if needed + if ( fontMetrics().width( text ) > 110 ) { + text.replace( ' ', "\n" ); + } +#endif + + setMinimumWidth( minimumSizeHint().width() ); + if ( parentWidget() ->minimumWidth() < minimumSizeHint().width() ) + parentWidget() ->setMinimumWidth( minimumSizeHint().width() + 10 ); + + update(); +} + +void KreMenuButton::mousePressEvent ( TQMouseEvent * ) +{ + emit clicked(); +} + +void KreMenuButton::rescale() +{ + resize( parentWidget() ->width() - 10, height() ); +} +TQSize KreMenuButton::sizeHint() const +{ + if ( parentWidget() ) + return ( TQSize( parentWidget() ->size().width() - 10, 40 ) ); + else + return TQSize( 100, 30 ); +} + +TQSize KreMenuButton::minimumSizeHint() const +{ + int text_width = TQMAX( fontMetrics().width( text.section( '\n', 0, 0 ) ), fontMetrics().width( text.section( '\n', 1, 1 ) ) ); + + if ( icon ) + return TQSize( 40 + icon->width() + text_width, 30 ); + else + return TQSize( 40 + text_width, 30 ); +} + +void KreMenuButton::paintEvent( TQPaintEvent * ) +{ + if ( !isShown() ) + return ; + // First draw the gradient + int darken = 130, lighten = 120; + TQColor c1, c2, c1h, c2h; //non-highlighted and highlighted versions + + // Set the gradient colors + + c1 = colorGroup().button().dark( darken ); + c2 = colorGroup().button().light( lighten ); + + if ( highlighted ) { + darken -= 10; + lighten += 10; + c1h = TDEGlobalSettings::highlightColor().dark( darken ); + c2h = TDEGlobalSettings::highlightColor().light( lighten ); + } + + // draw the gradient now + + TQPainter painter; + KPixmap kpm; + kpm.resize( ( ( TQWidget * ) parent() ) ->size() ); // use parent's same size to obtain the same gradient + + if ( !highlighted ) { + + // first the gradient + KPixmapEffect::unbalancedGradient ( kpm, c2, c1, KPixmapEffect::HorizontalGradient, -150, -150 ); + + } + else { + + // top gradient (highlighted) + kpm.resize( width(), height() ); + KPixmapEffect::unbalancedGradient ( kpm, c2h, c1h, KPixmapEffect::HorizontalGradient, -150, -150 ); + // low gradient besides the line (not hightlighted) + KPixmap kpmb; + kpmb.resize( width(), 2 ); + KPixmapEffect::unbalancedGradient ( kpmb, c2, c1, KPixmapEffect::HorizontalGradient, -150, -150 ); + // mix the two + bitBlt( &kpm, 0, height() - 2, &kpmb ); + + } + + // Draw the line + painter.begin( &kpm ); + painter.setPen( colorGroup().button().dark( darken ) ); + painter.drawLine( width() / 5, height() - 2, width() - 1, height() - 2 ); + painter.setPen( colorGroup().button().light( lighten ) ); + painter.drawLine( width() / 5, height() - 1, width() - 1, height() - 1 ); + painter.end(); + + + // Now Add the icon + + painter.begin( &kpm ); + int xPos, yPos; + if ( icon ) { + // Set the icon's desired horizontal position + + xPos = 10; + yPos = 0; + + + // Make sure it fits in the area + // If not, resize and reposition horizontally to be centered + + TQPixmap scaledIcon = *icon; + + if ( ( icon->height() > height() ) || ( icon->width() > width() / 3 ) ) // Nice effect, make sure you take less than half in width and fit in height (try making the menu very short in width) + { + TQImage image; + image = ( *icon ); + scaledIcon.convertFromImage( image.smoothScale( width() / 3, height(), TQImage::ScaleMin ) ); + } + + // Calculate the icon's vertical position + + yPos = ( height() - scaledIcon.height() ) / 2 - 1; + + + // Now draw it + + painter.drawPixmap( xPos, yPos, scaledIcon ); + + xPos += scaledIcon.width(); // increase it to place the text area correctly + } + + painter.end(); + + // If it's highlighted, draw a rounded area around the text + + // Calculate the rounded area + + int areax = xPos + 10; + int areah = fontMetrics().height() * ( text.contains( '\n' ) + 1 ) + fontMetrics().lineSpacing() * text.contains( '\n' ) + 6; // Make sure the area is sensible for text and adjust for multiple lines + + int areaw = width() - areax - 10; + + if ( areah > ( height() - 4 ) ) { + areah = height() - 4; // Limit to button height + } + + int areay = ( height() - areah - 2 ) / 2 + 1; // Center the area vertically + + + // Calculate roundness + + int roundy = 99, roundx = ( int ) ( ( float ) roundy * areah / areaw ); //Make corners round + + + if ( highlighted && areaw > 0 ) // If there is no space for the text area do not draw it + { + + // Draw the gradient + KPixmap area; + area.resize( areaw, areah ); + + + KPixmapEffect::gradient( area, c2h.light( 150 ), c1h.light( 150 ), KPixmapEffect::VerticalGradient ); + + painter.begin( &area ); + painter.setPen( c1h ); + painter.setBrush( TQt::NoBrush ); + painter.drawRoundRect( 0, 0, areaw, areah, roundx, roundy ); + painter.end(); + + // Make it round + TQBitmap mask( TQSize( areaw, areah ) ); + mask.fill( TQt::color0 ); + painter.begin( &mask ); + painter.setPen( TQt::color1 ); + painter.setBrush( TQt::color1 ); + painter.drawRoundRect( 0, 0, areaw, areah, roundx, roundy ); + painter.end(); + area.setMask( mask ); + + // Copy it to the button + bitBlt( &kpm, areax, areay, &area ); + } + + // Finally, draw the text besides the icon + TQRect r = rect(); + r.setLeft( areax + 5 ); + r.setWidth( areaw - 10 ); + + painter.begin( &kpm ); + if ( highlighted ) + painter.setPen( TDEGlobalSettings::highlightedTextColor() ); + else + painter.setPen( TDEGlobalSettings::textColor() ); + painter.setClipRect( r ); + painter.drawText( r, TQt::AlignVCenter, text ); + painter.end(); + + // Copy the offscreen button to the widget + bitBlt( this, 0, 0, &kpm, 0, 0, width(), height() ); // Copy the image with correct button size (button is already smaller than parent in width to leave space for the handle, so no need to use -10) + +} + +void KreMenuButton::setIconSet( const TQIconSet &is ) +{ + delete icon; + + icon = new TQPixmap( is.pixmap( TQIconSet::Small, TQIconSet::Normal, TQIconSet::On ) ); + + setMinimumWidth( minimumSizeHint().width() ); + if ( parentWidget() ->minimumWidth() < minimumSizeHint().width() ) + parentWidget() ->setMinimumWidth( minimumSizeHint().width() + 10 ); +} + +Menu::Menu( void ) +{ + childPos = 10; // Initial button is on top (10px), then keep scrolling down + widgetNumber = 0; // Initially we have no buttons + activeButton = 0; // Button that is highlighted +} + + +Menu::Menu( const Menu &m ) +{ + activeButton = m.activeButton; + childPos = m.childPos; + widgetNumber = m.widgetNumber; + + copyMap( positionList, m.positionList ); + copyMap( widgetList, m.widgetList ); +} + +Menu::~Menu( void ) +{} + +Menu& Menu::operator=( const Menu &m ) +{ + + activeButton = m.activeButton; + childPos = m.childPos; + widgetNumber = m.widgetNumber; + + copyMap( positionList, m.positionList ); + copyMap( widgetList, m.widgetList ); + + return *this; +} + + +void Menu::addButton( KreMenuButton* button ) +{ + button->move( 0, childPos ); + button->rescale(); + childPos += button->height(); + positionList[ button ] = widgetNumber; // Store index for this widget, and increment number + widgetList[ widgetNumber ] = button; // Store the button in the list (the inverse mapping of the previous one) + widgetNumber++; +} + +void Menu::copyMap( TQMap <int, KreMenuButton*> &destMap, const TQMap <int, KreMenuButton*> &origMap ) +{ + TQMap<int, KreMenuButton*>::ConstIterator it; + destMap.clear(); + for ( it = origMap.begin(); it != origMap.end(); ++it ) { + destMap[ it.key() ] = it.data(); + } +} + +void Menu::copyMap( TQMap <KreMenuButton*, int> &destMap, const TQMap <KreMenuButton*, int> &origMap ) +{ + TQMap<KreMenuButton*, int>::ConstIterator it; + destMap.clear(); + for ( it = origMap.begin(); it != origMap.end(); ++it ) { + destMap[ it.key() ] = it.data(); + } +} + +#include "kremenu.moc" diff --git a/src/widgets/kremenu.h b/src/widgets/kremenu.h new file mode 100644 index 0000000..4b02632 --- /dev/null +++ b/src/widgets/kremenu.h @@ -0,0 +1,170 @@ +/*************************************************************************** +* Copyright (C) 2003 by * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef KREMENU_H +#define KREMENU_H + +#include <tqbuttongroup.h> +#include <tqevent.h> +#include <tqiconset.h> +#include <tqmap.h> +#include <tqpushbutton.h> +#include <tqstring.h> + +#include "krecipesview.h" //for KrePanel enum + + +/** +* @author Unai Garro +* @author Bosselut Cyril +*/ + +class Menu; +class KreMenu; +class KreMenuButton; +typedef TQValueList <Menu>::Iterator MenuId; + + +class Menu +{ +public: + // Methods + + Menu( void ); + Menu( const Menu &m ); + ~Menu( void ); + void addButton( KreMenuButton *button ); + Menu& operator=( const Menu &m ); + + // Variables + + TQMap <KreMenuButton*, int> positionList; // Stores the indexes for the widgets + TQMap <int, KreMenuButton*> widgetList; // Stores the widgets for each position (just the inverse mapping) + KreMenuButton* activeButton; // Indicates which button is highlighted + int childPos; + int widgetNumber; +private: + // Methods + void copyMap( TQMap <int, KreMenuButton*> &destMap, const TQMap <int, KreMenuButton*> &origMap ); + void copyMap( TQMap <KreMenuButton*, int> &destMap, const TQMap <KreMenuButton*, int> &origMap ); +}; + + +class KreMenu : public TQWidget +{ + TQ_OBJECT +public: + KreMenu( TQWidget *parent = 0, const char *name = 0 ); + ~KreMenu(); + + MenuId createSubMenu( const TQString &title, const TQString &icon ); + MenuId mainMenu( void ) + { + return mainMenuId; + } + MenuId currentMenu( void ) + { + return currentMenuId; + } + TQSize sizeHint() const; + TQSize minimumSizeHint() const; + void resizeEvent( TQResizeEvent* e ); + void highlightButton( KreMenuButton *button ); + + +protected: + + virtual void paintEvent ( TQPaintEvent *e ); + virtual void childEvent ( TQChildEvent *e ); + virtual void keyPressEvent( TQKeyEvent *e ); + +private: + //Variables + TQValueList <Menu> menus; + MenuId mainMenuId; + MenuId currentMenuId; + Menu *m_currentMenu; + +signals: + void resized( int, int ); + void clicked( KrePanel ); + +private slots: + void collectClicks( KreMenuButton *w ); + void showMenu( MenuId id ); + +}; + +class KreMenuButton: public TQWidget +{ + TQ_OBJECT +public: + KreMenuButton( KreMenu *parent, KrePanel panel = KrePanel( -1 ), MenuId id = 0, const char *name = 0 ); + + ~KreMenuButton(); + + TQSize sizeHint() const; + TQSize minimumSizeHint() const; + + TQString title( void ) + { + return text; + } + void setActive( bool active = true ) + { + highlighted = active; + } + void setIconSet( const TQIconSet &is ); + MenuId menuId; + MenuId subMenuId; + + KrePanel getPanel() const + { + return panel; + } + +signals: + void resized( int, int ); + void clicked( void ); + void clicked( KreMenuButton* ); // sent together with clicked() + void clicked( MenuId ); // sent together with clicked() + +public slots: + void setTitle( const TQString &s ); + void rescale( void ); + +private: + // Button parts + TQPixmap* icon; + TQString text; + bool highlighted; + + KrePanel panel; + +private slots: + + void forwardClicks( void ) + { + emit clicked( this ); + if ( subMenuId != 0 ) + emit clicked( subMenuId ); + } + +protected: + + virtual void paintEvent( TQPaintEvent *e ); + virtual void mousePressEvent ( TQMouseEvent * e ); + +}; + + + +#endif diff --git a/src/widgets/kreruler.cpp b/src/widgets/kreruler.cpp new file mode 100644 index 0000000..448a375 --- /dev/null +++ b/src/widgets/kreruler.cpp @@ -0,0 +1,1049 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999 Reginald Stadlbauer <[email protected]> + Copyright (C) 2005 Jason Kivlighn <[email protected]> + + This library is free software; you can redistribute it and/or + modify it under the terms of the GNU Library General Public + License as published by the Free Software Foundation; either + version 2 of the License, or (at your option) any later version. + + This library 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 Library General Public License + along with this library; see the file COPYING.LIB. If not, write to + the Free Software Foundation, Inc., 59 Temple Place - Suite 330, + Boston, MA 02111-1307, USA. +*/ + +// Description: Ruler (header) + +/******************************************************************/ + +#include "kreruler.h" + +#include <tdelocale.h> +#include <tdeglobalsettings.h> +#include <kdebug.h> +#include <kiconloader.h> +#include <tqcursor.h> +#include <tqpainter.h> +#include <tqpopupmenu.h> +#include <tqtooltip.h> + +#include "krepagelayout.h" + +class KoRulerPrivate { +public: + KoRulerPrivate() { + } + ~KoRulerPrivate() {} + + TQWidget *canvas; + int flags; + int oldMx, oldMy; + bool whileMovingBorderLeft, whileMovingBorderRight; + bool whileMovingBorderTop, whileMovingBorderBottom; + TQPixmap pmFirst, pmLeft; + KoPageLayout layout; + KoTabulatorList tabList; + // Do we have to remove a certain tab in the DC Event? + KoTabulator removeTab; + // The tab we're moving / clicking on - basically only valid between press and release time + KoTabulator currTab; + // The action we're currently doing - basically only valid between press and release time + KoRuler::Action action; + TQPopupMenu *rb_menu; + int mRemoveTab, mPageLayout; // menu item ids + int frameEnd; + double i_right; + bool m_bReadWrite; + bool doubleClickedIndent; + bool rtl; + bool mousePressed; +}; + +// Equality test for tab positions in particular +static inline bool equals( double a, double b ) { + return kAbs( a - b ) < 1E-4; +} + + +/******************************************************************/ +/* Class: KoRuler */ +/******************************************************************/ + +const int KoRuler::F_TABS = 1; +const int KoRuler::F_INDENTS = 2; +const int KoRuler::F_HELPLINES = 4; +const int KoRuler::F_NORESIZE = 8; + +/*================================================================*/ +KoRuler::KoRuler( TQWidget *_parent, TQWidget *_canvas, Orientation _orientation, + const KoPageLayout& _layout, int _flags, KoUnit::Unit _unit ) + : TQFrame( _parent ), buffer( width(), height() ), m_zoom(1.0), m_1_zoom(1.0), + m_unit( _unit ) +{ + setWFlags( WResizeNoErase | WRepaintNoErase ); + setFrameStyle( MenuBarPanel ); + + d=new KoRulerPrivate(); + + d->canvas = _canvas; + orientation = _orientation; + d->layout = _layout; + d->flags = _flags; + + d->m_bReadWrite=true; + d->doubleClickedIndent=false; + diffx = 0; + diffy = 0; + i_left=0.0; + i_first=0.0; + d->i_right=0.0; + + setMouseTracking( true ); + d->mousePressed = false; + d->action = A_NONE; + + d->oldMx = 0; + d->oldMy = 0; + d->rtl = false; + + showMPos = false; + mposX = 0; + mposY = 0; + gridSize=0.0; + hasToDelete = false; + d->whileMovingBorderLeft = d->whileMovingBorderRight = d->whileMovingBorderTop = d->whileMovingBorderBottom = false; + + d->pmFirst = UserIcon( "koRulerFirst" ); + d->pmLeft = UserIcon( "koRulerLeft" ); + d->currTab.type = T_INVALID; + + d->removeTab.type = T_INVALID; + if ( orientation == TQt::Horizontal ) { + frameStart = tqRound( zoomIt(d->layout.ptLeft) ); + d->frameEnd = tqRound( zoomIt(d->layout.ptWidth - d->layout.ptRight) ); + } else { + frameStart = tqRound( zoomIt(d->layout.ptTop) ); + d->frameEnd = tqRound( zoomIt(d->layout.ptHeight - d->layout.ptBottom) ); + } + m_bFrameStartSet = false; + + setupMenu(); + + // For compatibility, emitting doubleClicked shall emit openPageLayoutDia + connect( this, TQ_SIGNAL( doubleClicked() ), this, TQ_SIGNAL( openPageLayoutDia() ) ); +} + +/*================================================================*/ +KoRuler::~KoRuler() +{ + delete d->rb_menu; + delete d; +} + +void KoRuler::setPageLayoutMenuItemEnabled(bool b) +{ + d->rb_menu->setItemEnabled(d->mPageLayout, b); +} + +/*================================================================*/ +void KoRuler::setMousePos( int mx, int my ) +{ + if ( !showMPos || ( mx == mposX && my == mposY ) ) return; + + TQPainter p( this ); + p.setRasterOp( TQt::NotROP ); + + if ( orientation == TQt::Horizontal ) { + if ( hasToDelete ) + p.drawLine( mposX, 1, mposX, height() - 1 ); + p.drawLine( mx, 1, mx, height() - 1 ); + hasToDelete = true; + } + else { + if ( hasToDelete ) + p.drawLine( 1, mposY, width() - 1, mposY ); + p.drawLine( 1, my, width() - 1, my ); + hasToDelete = true; + } + p.end(); + + mposX = mx; + mposY = my; +} + +// distance between the main lines (those with a number) +double KoRuler::lineDistance() const +{ + switch( m_unit ) { + case KoUnit::U_INCH: + return INCH_TO_POINT( m_zoom ); // every inch + case KoUnit::U_PT: + return 100.0 * m_zoom; // every 100 pt + case KoUnit::U_MM: + case KoUnit::U_CM: + case KoUnit::U_DM: + return CM_TO_POINT ( m_zoom ); // every cm + case KoUnit::U_PI: + return PI_TO_POINT ( 10.0 * m_zoom ); // every 10 pica + case KoUnit::U_DD: + return DD_TO_POINT( m_zoom ); // every diderot + case KoUnit::U_CC: + return CC_TO_POINT( 10.0 * m_zoom ); // every 10 cicero + } + // should never end up here + return 100.0 * m_zoom; +} + +/*================================================================*/ +void KoRuler::drawHorizontal( TQPainter *_painter ) +{ + TQFont font = TDEGlobalSettings::toolBarFont(); + TQFontMetrics fm( font ); + resize( width(), TQMAX( fm.height() + 4, 20 ) ); + + // Use a double-buffer pixmap + TQPainter p( &buffer ); + p.fillRect( 0, 0, width(), height(), TQBrush( colorGroup().brush( TQColorGroup::Background ) ) ); + + int totalw = tqRound( zoomIt(d->layout.ptWidth) ); + TQString str; + + p.setBrush( colorGroup().brush( TQColorGroup::Base ) ); + + // Draw white rect + TQRect r; + if ( !d->whileMovingBorderLeft ) + r.setLeft( -diffx + frameStart ); + else + r.setLeft( d->oldMx ); + r.setTop( 0 ); + if ( !d->whileMovingBorderRight ) + r.setWidth(d->frameEnd-frameStart); + else + r.setRight( d->oldMx ); + r.setBottom( height() ); + + p.drawRect( r ); + p.setFont( font ); + + // Draw the numbers + double dist = lineDistance(); + int maxwidth = 0; + + for ( double i = 0.0;i <= (double)totalw;i += dist ) { + str = TQString::number( KoUnit::toUserValue( i / m_zoom, m_unit ) ); + int textwidth = fm.width( str ); + maxwidth = TQMAX( maxwidth, textwidth ); + } + + // Make sure that the ruler stays readable at lower zoom levels + while( dist <= maxwidth ) { + dist += lineDistance(); + } + + for ( double i = 0.0;i <= (double)totalw;i += dist ) { + str = TQString::number( KoUnit::toUserValue( i / m_zoom, m_unit ) ); + int textwidth = fm.width( str ); + maxwidth = TQMAX( maxwidth, textwidth ); + p.drawText( tqRound(i) - diffx - tqRound(textwidth * 0.5), + tqRound(( height() - fm.height() ) * 0.5), + textwidth, height(), AlignLeft | AlignTop, str ); + } + + // Draw the medium-sized lines + // Only if we have enough space (i.e. not at 33%) + if ( dist > maxwidth + 2 ) + { + for ( double i = dist * 0.5;i <= (double)totalw;i += dist ) { + int ii=tqRound(i); + p.drawLine( ii - diffx, 7, ii - diffx, height() - 7 ); + } + } + + // Draw the small lines + // Only if we have enough space (i.e. not at 33%) + if ( dist * 0.5 > maxwidth + 2 ) + { + for ( double i = dist * 0.25;i <= (double)totalw;i += dist * 0.5 ) { + int ii=tqRound(i); + p.drawLine( ii - diffx, 9, ii - diffx, height() - 9 ); + } + } + + // Draw ending bar (at page width) + //int constant=zoomIt(1); + //p.drawLine( totalw - diffx + constant, 1, totalw - diffx + constant, height() - 1 ); + //p.setPen( colorGroup().color( TQColorGroup::Base ) ); + //p.drawLine( totalw - diffx, 1, totalw - diffx, height() - 1 ); + + // Draw starting bar (at 0) + //p.setPen( colorGroup().color( TQColorGroup::Text ) ); + //p.drawLine( -diffx, 1, -diffx, height() - 1 ); + //p.setPen( colorGroup().color( TQColorGroup::Base ) ); + //p.drawLine( -diffx - constant, 1, -diffx - constant, height() - 1 ); + + // Show the mouse position + if ( d->action == A_NONE && showMPos ) { + p.setPen( colorGroup().color( TQColorGroup::Text ) ); + p.drawLine( mposX, 1, mposX, height() - 1 ); + } + hasToDelete = false; + + p.end(); + _painter->drawPixmap( 0, 0, buffer ); +} + + +/*================================================================*/ +void KoRuler::drawVertical( TQPainter *_painter ) +{ + TQFont font = TDEGlobalSettings::toolBarFont(); + TQFontMetrics fm( font ); + resize( TQMAX( fm.height() + 4, 20 ), height() ); + + TQPainter p( &buffer ); + p.fillRect( 0, 0, width(), height(), TQBrush( colorGroup().brush( TQColorGroup::Background ) ) ); + + int totalh = tqRound( zoomIt(d->layout.ptHeight) ); + // Clip rect - this gives basically always a rect like (2,2,width-2,height-2) + TQRect paintRect = _painter->clipRegion( TQPainter::CoordPainter ).boundingRect(); + // Ruler rect + TQRect rulerRect( 0, -diffy, width(), totalh ); + + if ( paintRect.intersects( rulerRect ) ) { + TQString str; + + p.setBrush( colorGroup().brush( TQColorGroup::Base ) ); + + // Draw white rect + TQRect r; + if ( !d->whileMovingBorderTop ) + r.setTop( -diffy + frameStart ); + else + r.setTop( d->oldMy ); + r.setLeft( 0 ); + if ( !d->whileMovingBorderBottom ) + r.setHeight(d->frameEnd-frameStart); + else + r.setBottom( d->oldMy ); + r.setRight( width() ); + + p.drawRect( r ); + p.setFont( font ); + + // Draw the numbers + double dist = lineDistance(); + int maxheight = 0; + + for ( double i = 0.0;i <= (double)totalh;i += dist ) { + str = TQString::number( KoUnit::toUserValue( i / m_zoom, m_unit ) ); + int textwidth = fm.width( str ); + maxheight = TQMAX( maxheight, textwidth ); + } + + // Make sure that the ruler stays readable at lower zoom levels + while( dist <= maxheight ) { + dist += lineDistance(); + } + + for ( double i = 0.0;i <= (double)totalh;i += dist ) { + str = TQString::number( KoUnit::toUserValue( i / m_zoom, m_unit ) ); + int textheight = fm.height(); + int textwidth = fm.width( str ); + maxheight = TQMAX( maxheight, textwidth ); + p.save(); + p.translate( tqRound(( width() - textheight ) * 0.5), + tqRound(i) - diffy + tqRound(textwidth * 0.5) ); + p.rotate( -90 ); + p.drawText( 0, 0, textwidth + 1, textheight, AlignLeft | AlignTop, str ); + p.restore(); + } + + // Draw the medium-sized lines + if ( dist > maxheight + 2 ) + { + for ( double i = dist * 0.5;i <= (double)totalh;i += dist ) { + int ii=tqRound(i); + p.drawLine( 7, ii - diffy, width() - 7, ii - diffy ); + } + } + + // Draw the small lines + if ( dist * 0.5 > maxheight + 2 ) + { + for ( double i = dist * 0.25;i <=(double)totalh;i += dist *0.5 ) { + int ii=tqRound(i); + p.drawLine( 9, ii - diffy, width() - 9, ii - diffy ); + } + } + + // Draw ending bar (at page height) + //p.drawLine( 1, totalh - diffy + 1, width() - 1, totalh - diffy + 1 ); + //p.setPen( colorGroup().color( TQColorGroup::Base ) ); + //p.drawLine( 1, totalh - diffy, width() - 1, totalh - diffy ); + + // Draw starting bar (at 0) + //p.setPen( colorGroup().color( TQColorGroup::Text ) ); + //p.drawLine( 1, -diffy, width() - 1, -diffy ); + //p.setPen( colorGroup().color( TQColorGroup::Base ) ); + //p.drawLine( 1, -diffy - 1, width() - 1, -diffy - 1 ); + } + + // Show the mouse position + if ( d->action == A_NONE && showMPos ) { + p.setPen( colorGroup().color( TQColorGroup::Text ) ); + p.drawLine( 1, mposY, width() - 1, mposY ); + } + hasToDelete = false; + + p.end(); + _painter->drawPixmap( 0, 0, buffer ); +} + +void KoRuler::mousePressEvent( TQMouseEvent *e ) +{ + if( !d->m_bReadWrite) + return; + + d->oldMx = e->x(); + d->oldMy = e->y(); + d->mousePressed = true; + d->removeTab.type = T_INVALID; + + switch ( e->button() ) { + case RightButton: + if(d->currTab.type == T_INVALID || !(d->flags & F_TABS)) + d->rb_menu->setItemEnabled(d->mRemoveTab, false); + else + d->rb_menu->setItemEnabled(d->mRemoveTab, true); + d->rb_menu->popup( TQCursor::pos() ); + d->action = A_NONE; + d->mousePressed = false; + return; + case MidButton: + // MMB shall do like double-click (it opens a dialog). + handleDoubleClick(); + return; + case LeftButton: + if ( d->action == A_BR_RIGHT || d->action == A_BR_LEFT ) { + if ( d->action == A_BR_RIGHT ) + d->whileMovingBorderRight = true; + else + d->whileMovingBorderLeft = true; + + if ( d->canvas ) + drawLine(d->oldMx, -1); + update(); + } else if ( d->action == A_BR_TOP || d->action == A_BR_BOTTOM ) { + if ( d->action == A_BR_TOP ) + d->whileMovingBorderTop = true; + else + d->whileMovingBorderBottom = true; + + if ( d->canvas ) { + TQPainter p( d->canvas ); + p.setRasterOp( TQt::NotROP ); + p.drawLine( 0, d->oldMy, d->canvas->width(), d->oldMy ); + p.end(); + } + update(); + } else if ( d->action == A_FIRST_INDENT || d->action == A_LEFT_INDENT || d->action == A_RIGHT_INDENT ) { + if ( d->canvas ) + drawLine(d->oldMx, -1); + } else if ( d->action == A_TAB ) { + if ( d->canvas && d->currTab.type != T_INVALID ) { + drawLine( tqRound( applyRtlAndZoom(d->currTab.ptPos) ) + frameStart - diffx, -1 ); + } + } + else if ( d->flags & F_HELPLINES ) + { + setCursor( orientation == TQt::Horizontal ? + TQt::sizeVerCursor : TQt::sizeHorCursor ); + d->action = A_HELPLINES; + } + default: + break; + } +} + +void KoRuler::mouseReleaseEvent( TQMouseEvent *e ) +{ + d->mousePressed = false; + + // Hacky, but necessary to prevent multiple tabs with the same coordinates (Werner) + bool fakeMovement=false; + if(d->removeTab.type != T_INVALID) { + mouseMoveEvent(e); + fakeMovement=true; + } + + if ( d->action == A_BR_RIGHT || d->action == A_BR_LEFT ) { + d->whileMovingBorderRight = false; + d->whileMovingBorderLeft = false; + + if ( d->canvas ) + drawLine(d->oldMx, -1); + update(); + emit newPageLayout( d->layout ); + } else if ( d->action == A_BR_TOP || d->action == A_BR_BOTTOM ) { + d->whileMovingBorderTop = false; + d->whileMovingBorderBottom = false; + + if ( d->canvas ) { + TQPainter p( d->canvas ); + p.setRasterOp( TQt::NotROP ); + p.drawLine( 0, d->oldMy, d->canvas->width(), d->oldMy ); + p.end(); + } + update(); + emit newPageLayout( d->layout ); + } else if ( d->action == A_FIRST_INDENT ) { + if ( d->canvas ) + drawLine(d->oldMx, -1); + update(); + emit newFirstIndent( i_first ); + } else if ( d->action == A_LEFT_INDENT ) { + if ( d->canvas ) + drawLine(d->oldMx, -1); + update(); + emit newLeftIndent( i_left ); + } else if ( d->action == A_RIGHT_INDENT ) { + if ( d->canvas ) + drawLine(d->oldMx, -1); + update(); + emit newRightIndent( d->i_right ); + } else if ( d->action == A_TAB ) { + if ( d->canvas && !fakeMovement ) { + drawLine( tqRound( applyRtlAndZoom( d->currTab.ptPos ) ) + frameStart - diffx, -1); + } + if ( willRemoveTab( e->y() ) ) + { + d->tabList.remove(d->currTab); + } + qHeapSort( d->tabList ); + + // Delete the new tabulator if it is placed on top of another. + KoTabulatorList::ConstIterator tmpTab=d->tabList.begin(); + int count=0; + while(tmpTab!=d->tabList.end()) { + if( equals( (*tmpTab).ptPos, d->currTab.ptPos ) ) { + count++; + if(count > 1) { + d->tabList.remove(d->currTab); + break; + } + } + tmpTab++; + } + //searchTab( e->x() ); // DF: why set currTab here? + emit tabListChanged( d->tabList ); + update(); + } + else if( d->action == A_HELPLINES ) + { + emit addHelpline( e->pos(), orientation == TQt::Horizontal); + setCursor( ArrowCursor ); + } + d->currTab.type = T_INVALID; // added (DF) +} + +void KoRuler::mouseMoveEvent( TQMouseEvent *e ) +{ + hasToDelete = false; + + int pw = d->frameEnd - frameStart; + int ph = tqRound(zoomIt(d->layout.ptHeight)); + int left = frameStart - diffx; + int top = tqRound(zoomIt(d->layout.ptTop)); + top -= diffy; + int right = d->frameEnd - diffx; + int bottom = tqRound(zoomIt(d->layout.ptBottom)); + bottom = ph - bottom - diffy; + // Cumulate first-line-indent + int ip_first = tqRound( zoomIt( i_first + ( d->rtl ? d->i_right : i_left) ) ); + int ip_left = tqRound(zoomIt(i_left)); + int ip_right = tqRound(zoomIt(d->i_right)); + + int mx = e->x(); + mx = mx+diffx < 0 ? 0 : mx; + int my = e->y(); + my = my+diffy < 0 ? 0 : my; + + TQToolTip::remove( this); + switch ( orientation ) { + case TQt::Horizontal: { + if ( !d->mousePressed ) { + setCursor( ArrowCursor ); + d->action = A_NONE; + /////// ###### + // At the moment, moving the left and right border indicators + // is disabled when setFrameStartEnd has been called (i.e. in KWord) + // Changing the layout margins directly from it would be utterly wrong + // (just try the 2-columns modes...). What needs to be done is: + // emitting a signal frameResized in mouseReleaseEvent, when a left/right + // border has been moved, and in kword we need to update the margins from + // there, if the left border of the 1st column or the right border of the + // last column was moved... and find what to do with the other borders. + // And for normal frames, resize the frame without touching the page layout. + // All that is too much for now -> disabling. + if ( !m_bFrameStartSet ) + { + if ( mx > left - 5 && mx < left + 5 ) { + setCursor( TQt::sizeHorCursor ); + d->action = A_BR_LEFT; + } else if ( mx > right - 5 && mx < right + 5 ) { + setCursor( TQt::sizeHorCursor ); + d->action = A_BR_RIGHT; + } + } + if ( d->flags & F_TABS ) + searchTab(mx); + } else { + // Calculate the new value. + int newPos=mx; + if( newPos!=right && gridSize!=0.0 && (e->state() & ShiftButton)==0) { // apply grid. + double grid=zoomIt(gridSize * 16); + newPos=tqRound( ((newPos * 16 / grid) * grid) / 16 ); + } + if(newPos-left < 0) newPos=left; + else if (right-newPos < 0) newPos=right; + double newValue = unZoomIt(static_cast<double>(newPos) - frameStart + diffx); + + switch ( d->action ) { + case A_BR_LEFT: { + if ( d->canvas && mx < right-10 && mx+diffx-2 > 0) { + drawLine( d->oldMx, mx ); + d->layout.ptLeft = unZoomIt(static_cast<double>(mx + diffx)); + if( ip_left > right-left-15 ) { + ip_left=right-left-15; + ip_left=ip_left<0 ? 0 : ip_left; + i_left=unZoomIt( ip_left ); + emit newLeftIndent( i_left ); + } + if ( ip_right > right-left-15 ) { + ip_right=right-left-15; + ip_right=ip_right<0? 0 : ip_right; + d->i_right=unZoomIt( ip_right ); + emit newRightIndent( d->i_right ); + } + d->oldMx = mx; + d->oldMy = my; + update(); + } + else + return; + } break; + case A_BR_RIGHT: { + if ( d->canvas && mx > left+10 && mx+diffx <= pw-2) { + drawLine( d->oldMx, mx ); + d->layout.ptRight = unZoomIt(static_cast<double>(pw - ( mx + diffx ))); + if( ip_left > right-left-15 ) { + ip_left=right-left-15; + ip_left=ip_left<0 ? 0 : ip_left; + i_left=unZoomIt( ip_left ); + emit newLeftIndent( i_left ); + } + if ( ip_right > right-left-15 ) { + ip_right=right-left-15; + ip_right=ip_right<0? 0 : ip_right; + d->i_right=unZoomIt( ip_right ); + emit newRightIndent( d->i_right ); + } + d->oldMx = mx; + d->oldMy = my; + update(); + } + else + return; + } break; + case A_FIRST_INDENT: { + if ( d->canvas ) { + if (d->rtl) + newValue = unZoomIt(pw) - newValue - d->i_right; + else + newValue -= i_left; + if(newValue == i_first) break; + drawLine( d->oldMx, newPos); + d->oldMx=newPos; + i_first = newValue; + update(); + } + } break; + case A_LEFT_INDENT: { + if ( d->canvas ) { + //if (d->rtl) newValue = unZoomIt(pw) - newValue; + if(newValue == i_left) break; + + drawLine( d->oldMx, newPos); + i_left = newValue; + d->oldMx = newPos; + update(); + } + } break; + case A_RIGHT_INDENT: { + if ( d->canvas ) { + double rightValue = unZoomIt(right - newPos); + //if (d->rtl) rightValue = unZoomIt(pw) - rightValue; + if(rightValue == d->i_right) break; + + drawLine( d->oldMx, newPos); + d->i_right=rightValue; + d->oldMx = newPos; + update(); + } + } break; + case A_TAB: { + if ( d->canvas) { + if (d->rtl) newValue = unZoomIt(pw) - newValue; + if(newValue == d->currTab.ptPos) break; // no change + TQPainter p( d->canvas ); + p.setRasterOp( TQt::NotROP ); + // prevent 1st drawLine when we just created a new tab + // (it's a NOT line) + double pt; + int pt_fr; + if( d->currTab != d->removeTab ) + { + pt = applyRtlAndZoom(d->currTab.ptPos); + pt_fr = tqRound(pt) + frameStart - diffx; + p.drawLine( pt_fr, 0, pt_fr, d->canvas->height() ); + } + + KoTabulatorList::Iterator it = d->tabList.find( d->currTab ); + Q_ASSERT( it != d->tabList.end() ); + if ( it != d->tabList.end() ) + (*it).ptPos = newValue; + d->currTab.ptPos = newValue; + + pt = applyRtlAndZoom( newValue ); + pt_fr = tqRound(pt) + frameStart - diffx; + p.drawLine( pt_fr, 0, pt_fr, d->canvas->height() ); + + p.end(); + d->oldMx = mx; + d->oldMy = my; + d->removeTab.type = T_INVALID; + update(); + } + } break; + default: break; + } + } + if( d->action == A_HELPLINES ) + { + emit moveHelpLines( e->pos(), orientation == TQt::Horizontal); + } + + return; + } break; + case TQt::Vertical: { + if ( !d->mousePressed ) { + setCursor( ArrowCursor ); + d->action = A_NONE; + if ( d->flags & F_NORESIZE ) + break; + if ( my > top - 5 && my < top + 5 ) { + TQToolTip::add( this, i18n("Top margin") ); + setCursor( TQt::sizeVerCursor ); + d->action = A_BR_TOP; + } else if ( my > bottom - 5 && my < bottom + 5 ) { + TQToolTip::add( this, i18n("Bottom margin") ); + setCursor( TQt::sizeVerCursor ); + d->action = A_BR_BOTTOM; + } + } else { + switch ( d->action ) { + case A_BR_TOP: { + if ( d->canvas && my < bottom-20 && my+diffy-2 > 0) { + TQPainter p( d->canvas ); + p.setRasterOp( TQt::NotROP ); + p.drawLine( 0, d->oldMy, d->canvas->width(), d->oldMy ); + p.drawLine( 0, my, d->canvas->width(), my ); + p.end(); + d->layout.ptTop = unZoomIt(static_cast<double>(my + diffy)); + d->oldMx = mx; + d->oldMy = my; + update(); + } + else + return; + } break; + case A_BR_BOTTOM: { + if ( d->canvas && my > top+20 && my+diffy < ph-2) { + TQPainter p( d->canvas ); + p.setRasterOp( TQt::NotROP ); + p.drawLine( 0, d->oldMy, d->canvas->width(), d->oldMy ); + p.drawLine( 0, my, d->canvas->width(), my ); + p.end(); + d->layout.ptBottom = unZoomIt(static_cast<double>(ph - ( my + diffy ))); + d->oldMx = mx; + d->oldMy = my; + update(); + } + else + return; + } break; + default: break; + } + } + } break; + } + if( d->action == A_HELPLINES ) + { + emit moveHelpLines( e->pos(), orientation == TQt::Horizontal); + } + + d->oldMx = mx; + d->oldMy = my; +} + +void KoRuler::resizeEvent( TQResizeEvent *e ) +{ + TQFrame::resizeEvent( e ); + buffer.resize( size() ); +} + +void KoRuler::mouseDoubleClickEvent( TQMouseEvent* ) +{ + handleDoubleClick(); +} + +void KoRuler::handleDoubleClick() +{ + if ( !d->m_bReadWrite ) + return; + + d->doubleClickedIndent = false; + + // When Binary Compatibility is broken this will hopefully emit a + // doubleClicked(int) to differentiate between double-clicking an + // indent and double-clicking the ruler + if ( d->flags & F_INDENTS ) { + if ( d->action == A_LEFT_INDENT || d->action == A_RIGHT_INDENT || d->action == A_FIRST_INDENT ) { + d->doubleClickedIndent = true; + emit doubleClicked(); // usually paragraph dialog + return; + } + } + + // Double-clicked nothing + d->action = A_NONE; + emit doubleClicked(); // usually page layout dialog +} + +void KoRuler::setTabList( const KoTabulatorList & _tabList ) +{ + d->tabList = _tabList; + qHeapSort(d->tabList); // "Trust no one." as opposed to "In David we trust." + + // Note that d->currTab and d->removeTab could now point to + // tabs which don't exist in d->tabList + + update(); +} + +double KoRuler::makeIntern( double _v ) +{ + return KoUnit::fromUserValue( _v, m_unit ); +} + +void KoRuler::setupMenu() +{ + d->rb_menu = new TQPopupMenu(); + TQ_CHECK_PTR( d->rb_menu ); + for ( uint i = 0 ; i <= KoUnit::U_LASTUNIT ; ++i ) + { + KoUnit::Unit unit = static_cast<KoUnit::Unit>( i ); + d->rb_menu->insertItem( KoUnit::unitDescription( unit ), i /*as id*/ ); + if ( m_unit == unit ) + d->rb_menu->setItemChecked( i, true ); + } + connect( d->rb_menu, TQ_SIGNAL( activated( int ) ), TQ_SLOT( slotMenuActivated( int ) ) ); + + d->rb_menu->insertSeparator(); + d->mPageLayout=d->rb_menu->insertItem(i18n("Page Layout..."), this, TQ_SLOT(pageLayoutDia())); +#if 0 + d->rb_menu->insertSeparator(); + d->mRemoveTab=d->rb_menu->insertItem(i18n("Remove Tabulator"), this, TQ_SLOT(rbRemoveTab())); + d->rb_menu->setItemEnabled( d->mRemoveTab, false ); +#endif +} + +void KoRuler::uncheckMenu() +{ + for ( uint i = 0 ; i <= KoUnit::U_LASTUNIT ; ++i ) + d->rb_menu->setItemChecked( i, false ); +} + +void KoRuler::setUnit( KoUnit::Unit unit ) +{ + m_unit = unit; + uncheckMenu(); + d->rb_menu->setItemChecked( m_unit, true ); + update(); +} + +void KoRuler::setZoom( const double& zoom ) +{ + if(zoom==m_zoom) + return; + if(zoom < 1E-4) // Don't do 0 or negative values + return; + m_zoom=zoom; + m_1_zoom=1/m_zoom; + update(); +} + +bool KoRuler::willRemoveTab( int y ) const +{ + return (y < -50 || y > height() + 25) && d->currTab.type != T_INVALID; +} + +void KoRuler::rbRemoveTab() { + + d->tabList.remove( d->currTab ); + d->currTab.type = T_INVALID; + emit tabListChanged( d->tabList ); + update(); +} + +void KoRuler::setReadWrite(bool _readWrite) +{ + d->m_bReadWrite=_readWrite; +} + +void KoRuler::searchTab(int mx) { + + int pos; + d->currTab.type = T_INVALID; + KoTabulatorList::ConstIterator it = d->tabList.begin(); + for ( ; it != d->tabList.end() ; ++it ) { + pos = tqRound(applyRtlAndZoom((*it).ptPos)) - diffx + frameStart; + if ( mx > pos - 5 && mx < pos + 5 ) { + setCursor( TQt::sizeHorCursor ); + d->action = A_TAB; + d->currTab = *it; + break; + } + } +} + +void KoRuler::drawLine(int oldX, int newX) { + + TQPainter p( d->canvas ); + p.setRasterOp( TQt::NotROP ); + p.drawLine( oldX, 0, oldX, d->canvas->height() ); + if(newX!=-1) + p.drawLine( newX, 0, newX, d->canvas->height() ); + p.end(); +} + +void KoRuler::showMousePos( bool _showMPos ) +{ + showMPos = _showMPos; + hasToDelete = false; + mposX = -1; + mposY = -1; + update(); +} + +void KoRuler::setOffset( int _diffx, int _diffy ) +{ + //kdDebug() << "KoRuler::setOffset " << _diffx << "," << _diffy << endl; + diffx = _diffx; + diffy = _diffy; + update(); +} + +void KoRuler::setFrameStartEnd( int _frameStart, int _frameEnd ) +{ + if ( _frameStart != frameStart || _frameEnd != d->frameEnd || !m_bFrameStartSet ) + { + frameStart = _frameStart; + d->frameEnd = _frameEnd; + // Remember that setFrameStartEnd was called. This activates a slightly + // different mode (when moving start and end positions). + m_bFrameStartSet = true; + update(); + } +} + +void KoRuler::setRightIndent( double _right ) +{ + d->i_right = makeIntern( _right ); + update(); +} + +void KoRuler::setDirection( bool rtl ) +{ + d->rtl = rtl; + update(); +} + +void KoRuler::changeFlags(int _flags) +{ + d->flags = _flags; +} + +int KoRuler::flags() const +{ + return d->flags; +} + +bool KoRuler::doubleClickedIndent() const +{ + return d->doubleClickedIndent; +} + +double KoRuler::applyRtlAndZoom( double value ) const +{ + int frameWidth = d->frameEnd - frameStart; + return d->rtl ? ( frameWidth - zoomIt( value ) ) : zoomIt( value ); +} + +double KoRuler::unZoomItRtl( int pixValue ) const +{ + int frameWidth = d->frameEnd - frameStart; + return d->rtl ? ( unZoomIt( (double)(frameWidth - pixValue) ) ) : unZoomIt( (double)pixValue ); +} + +void KoRuler::slotMenuActivated( int i ) +{ + if ( i >= 0 && i <= KoUnit::U_LASTUNIT ) + { + KoUnit::Unit unit = static_cast<KoUnit::Unit>(i); + setUnit( unit ); + emit unitChanged( unit ); + } +} + +TQSize KoRuler::minimumSizeHint() const +{ + TQSize size; + TQFont font = TDEGlobalSettings::toolBarFont(); + TQFontMetrics fm( font ); + + size.setWidth( TQMAX( fm.height() + 4, 20 ) ); + size.setHeight( TQMAX( fm.height() + 4, 20 ) ); + + return size; +} + +TQSize KoRuler::sizeHint() const +{ + return minimumSizeHint(); +} + +void KoRuler::setPageLayout( const KoPageLayout& _layout ) +{ + d->layout = _layout; + update(); +} + +#include "kreruler.moc" diff --git a/src/widgets/kreruler.h b/src/widgets/kreruler.h new file mode 100644 index 0000000..ccf76f2 --- /dev/null +++ b/src/widgets/kreruler.h @@ -0,0 +1,366 @@ +/* This file is part of the KDE project + Copyright (C) 1998, 1999 Reginald Stadlbauer <[email protected]> + Copyright (C) 2005 Jason Kivlighn <[email protected]> + + This library is free software; you can redistribute it and/or + modify it under the terms of the GNU Library General Public + License as published by the Free Software Foundation; either + version 2 of the License, or (at your option) any later version. + + This library 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 Library General Public License + along with this library; see the file COPYING.LIB. If not, write to + the Free Software Foundation, Inc., 59 Temple Place - Suite 330, + Boston, MA 02111-1307, USA. +*/ + +// Description: Ruler (header) + +/******************************************************************/ + +#ifndef KRERULER_H +#define KRERULER_H + +#include <tqframe.h> +#include <tqpixmap.h> + +#include <tdemacros.h> + +#if 0 +#include <koGlobal.h> +#include <koTabChooser.h> +#endif + +#include "kreunit.h" + +class KoPageLayout; +class TQPainter; + +enum KoTabulators { T_LEFT = 0, T_CENTER = 1, T_RIGHT = 2, T_DEC_PNT = 3, T_INVALID = -1 }; +enum KoTabulatorFilling { TF_BLANK = 0, TF_DOTS = 1, TF_LINE = 2, TF_DASH = 3, TF_DASH_DOT = 4, TF_DASH_DOT_DOT = 5}; + +/** + * Struct: KoTabulator + * Defines the position of a tabulation (in pt), and its type + */ +struct KoTabulator { + /** + * Position of the tab in pt + */ + double ptPos; + /** + * Type of tab (left/center/right/decimalpoint) + */ + KoTabulators type; + /** + * Type of tab filling. + */ + KoTabulatorFilling filling; + /** + * Width of the tab filling line. + */ + double ptWidth; + /** + * Alignment character. + */ + TQChar alignChar; + + bool operator==( const KoTabulator & t ) const { + return TQABS( ptPos - t.ptPos ) < 1E-4 && type == t.type && + filling == t.filling && TQABS( ptWidth - t.ptWidth ) < 1E-4; + } + bool operator!=( const KoTabulator & t ) const { + return !operator==(t); + } + // Operators used for sorting + bool operator < ( const KoTabulator & t ) const { + return ptPos < t.ptPos; + } + bool operator <= ( const KoTabulator & t ) const { + return ptPos <= t.ptPos; + } + bool operator > ( const KoTabulator & t ) const { + return ptPos > t.ptPos; + } +}; + +typedef TQValueList<KoTabulator> KoTabulatorList; + +class KoRulerPrivate; + +/** + * KoRuler is the horizontal or vertical ruler, to be used around + * the drawing area of most KOffice programs. + * + * It shows the graduated ruler with numbering, in any of the base units (like mm/pt/inch), + * and supports zooming, tabulators, paragraph indents, showing the mouse position, etc. + * + * It also offers a popupmenu upon right-clicking, for changing the unit, + * the page layout, or removing a tab. + */ +class KoRuler : public TQFrame +{ + TQ_OBJECT + friend class KoRulerPrivate; // for the Action enum +public: + static const int F_TABS; + static const int F_INDENTS; + static const int F_HELPLINES; + static const int F_NORESIZE; + + /** + * Create a ruler + * TODO document params + */ + KoRuler( TQWidget *_parent, TQWidget *_canvas, Orientation _orientation, + const KoPageLayout& _layout, int _flags, KoUnit::Unit _unit ); + ~KoRuler(); + + /** + * Set the unit to be used. + */ + void setUnit( KoUnit::Unit unit ); + + /** + * Set the zoom of the ruler (default value of 1.0 is 100%) + */ + void setZoom( const double& zoom=1.0 ); + /** + * @return the current zoom level + */ + const double& zoom() const { return m_zoom; } + + /** + * Set the page layout, see @ref KoPageLayout. + * This defines the size of the page and the margins, + * from which the size of the ruler is deducted. + */ + void setPageLayout( const KoPageLayout& _layout ); + + /** + * Call showMousePos(true) if the ruler should indicate the position + * of the mouse. This is usually only the case for drawing applications, + * so it is not enabled by default. + */ + void showMousePos( bool _showMPos ); + /** + * Set the position of the mouse, to update the indication in the ruler. + * This is only effective if showMousePos(true) was called previously. + * The position to give is not zoomed, it's in real pixel coordinates! + */ + void setMousePos( int mx, int my ); + + /** + * Set a global offset to the X and Y coordinates. + * Usually the main drawing area is a TQScrollView, and this is called + * with contentsX() and contentsY(), each time those values change. + */ + void setOffset( int _diffx, int _diffy ); + + /** + * Set the [paragraph] left indent to the specified position (in the current unit) + */ + void setLeftIndent( double _left ) + { i_left = makeIntern( _left ); update(); } + + /** + * Set the [paragraph] first-line left indent to the specified position (in the current unit) + * This indent is cumulated with the left or right margin, depending on the [paragraph] direction. + */ + void setFirstIndent( double _first ) + { i_first = makeIntern( _first ); update(); } + + /** + * Set the [paragraph] right indent to the specified position (in the current unit) + */ + void setRightIndent( double _right ); + + /** + * Set the [paragraph] direction. By default (rtl=false), the left indent is on the + * left, and the right indent is on the right ;) + * If rtl=true, it's the opposite. + */ + void setDirection( bool rtl ); + + /** + * Set the list of tabulators to show in the ruler. + */ + void setTabList( const KoTabulatorList & tabList ); + + /** + * Set the start and the end of the current 'frame', i.e. the part + * of the page in which we are currently working. See KWord frames + * for an example where this is used. The tab positions and paragraph + * indents then become relative to the beginning of the frame, and the + * ruler goes from frameStart to frameEnd instead of using the page margins. + * @p _frameStart et @p _frameEnd are in pixel coordinates. + */ + void setFrameStartEnd( int _frameStart, int _frameEnd ); + + /** + * KoRuler is in "read write" mode by default. + * Use setReadWrite(false) to use it in read-only mode. + */ + void setReadWrite( bool _readWrite ); + + /** + * Change the flag (i.e. activate or deactivate certain features of KoRuler) + */ + void changeFlags(int _flags); + + /** + * Set the size of the grid used for tabs positioning, size in pt. + * default value is 0. 0 means no grid. + */ + void setGridSize(double newGridSize) { gridSize=newGridSize; } + + /** + * @return the current flags + */ + int flags() const; + + /** + * @return whether the current doubleClicked() signal was triggered + * by the user double-clicking on an indent (BCI). It returns false + * if the user just double-clicked on an "empty" part of the ruler. + * + * This method is strictly provided for use in a slot connected to the + * doubleClicked() signal; calling it at any other time results in + * undefined behavior. + */ + bool doubleClickedIndent() const; + + /** + * Enable or disable the "Page Layout" menu item. + */ + void setPageLayoutMenuItemEnabled(bool b); + + /** + * Reimplemented from TQWidget + */ + virtual TQSize minimumSizeHint() const; + + /** + * Reimplemented from TQWidget + */ + virtual TQSize sizeHint() const; + +signals: + void newPageLayout( const KoPageLayout & ); + void newLeftIndent( double ); + void newFirstIndent( double ); + void newRightIndent( double ); + /** Old signal, kept for compatibility. Use doubleClicked instead. */ + void openPageLayoutDia(); + /** This signal is emitted when double-clicking the ruler (or an indent) */ + void doubleClicked(); + /** This signal is emitted when double-clicking a tab */ + void doubleClicked( double ptPos ); + + void tabListChanged( const KoTabulatorList & ); + void unitChanged( KoUnit::Unit ); + + void addHelpline(const TQPoint &, bool ); + void moveHelpLines( const TQPoint &, bool ); + +protected: + enum Action {A_NONE, A_BR_LEFT, A_BR_RIGHT, A_BR_TOP, A_BR_BOTTOM, + A_LEFT_INDENT, A_FIRST_INDENT, A_TAB, A_RIGHT_INDENT, + A_HELPLINES }; + + void drawContents( TQPainter *_painter ) + { orientation == TQt::Horizontal ? drawHorizontal( _painter ) : drawVertical( _painter ); } + + void drawHorizontal( TQPainter *_painter ); + void drawVertical( TQPainter *_painter ); + + void mousePressEvent( TQMouseEvent *e ); + void mouseReleaseEvent( TQMouseEvent *e ); + void mouseMoveEvent( TQMouseEvent *e ); + void mouseDoubleClickEvent( TQMouseEvent* ); + void resizeEvent( TQResizeEvent *e ); + void handleDoubleClick(); + + double makeIntern( double _v ); + double zoomIt(const double &value) const; + int zoomIt(const int &value) const; + unsigned int zoomIt(const unsigned int &value) const; + double unZoomIt(const double &value) const; + int unZoomIt(const int &value) const; + unsigned int unZoomIt(const unsigned int &value) const; + void setupMenu(); + void uncheckMenu(); + void searchTab(int mx); + void drawLine(int oldX, int newX); + +private: + double applyRtlAndZoom( double value ) const; + double unZoomItRtl( int pixValue ) const; + double lineDistance() const; + bool willRemoveTab( int y ) const; + + KoRulerPrivate *d; + + TQt::Orientation orientation; + int diffx, diffy; + double i_left, i_first; + TQPixmap buffer; + double m_zoom, m_1_zoom; + KoUnit::Unit m_unit; + bool hasToDelete; + bool showMPos; + bool m_bFrameStartSet; + bool m_bReadWrite; + int mposX, mposY; + int frameStart; + + double gridSize; + +protected slots: + void slotMenuActivated( int i ); + void pageLayoutDia() { emit doubleClicked()/*openPageLayoutDia()*/; } + void rbRemoveTab(); + +}; + +inline double KoRuler::zoomIt(const double &value) const { + if (m_zoom==1.0) + return value; + return m_zoom*value; +} + +inline int KoRuler::zoomIt(const int &value) const { + if (m_zoom==1.0) + return value; + return tqRound(m_zoom*value); +} + +inline unsigned int KoRuler::zoomIt(const unsigned int &value) const { + if (m_zoom==1.0) + return value; + return static_cast<unsigned int>(tqRound(m_zoom*value)); +} + +inline double KoRuler::unZoomIt(const double &value) const { + if(m_zoom==1.0) + return value; + return value*m_1_zoom; +} + +inline int KoRuler::unZoomIt(const int &value) const { + if(m_zoom==1.0) + return value; + return tqRound(value*m_1_zoom); +} + +inline unsigned int KoRuler::unZoomIt(const unsigned int &value) const { + if(m_zoom==1.0) + return value; + return static_cast<unsigned int>(tqRound(value*m_1_zoom)); +} + +#endif diff --git a/src/widgets/kretextedit.cpp b/src/widgets/kretextedit.cpp new file mode 100644 index 0000000..c6d150d --- /dev/null +++ b/src/widgets/kretextedit.cpp @@ -0,0 +1,183 @@ +/*************************************************************************** +* Copyright (C) 2004 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "kretextedit.h" + +#include <tqtextstream.h> + +#include <tdeaccel.h> +#include <kdebug.h> + +KreTextEdit::KreTextEdit( TQWidget *parent ) : KTextEdit( parent ), TDECompletionBase() +{ + TDECompletion * comp = completionObject(); //creates the completion object + comp->setIgnoreCase( true ); + + completing = false; + + connect( this, TQ_SIGNAL( clicked( int, int ) ), TQ_SLOT( haltCompletion() ) ); +} + +void KreTextEdit::haltCompletion() +{ + completing = false; +} + +void KreTextEdit::keyPressEvent( TQKeyEvent *e ) +{ + // Filter key-events if completion mode is not set to CompletionNone + KKey key( e ); + + KeyBindingMap keys = getKeyBindings(); + TDEShortcut cut; + bool noModifier = ( e->state() == NoButton || e->state() == ShiftButton ); + + if ( noModifier ) { + TQString keycode = e->text(); + if ( !keycode.isEmpty() && keycode.unicode() ->isPrint() ) { + TQTextEdit::keyPressEvent ( e ); + tryCompletion(); + e->accept(); + return ; + } + } + + // Handles completion + if ( keys[ TextCompletion ].isNull() ) + cut = TDEStdAccel::shortcut( TDEStdAccel::TextCompletion ); + else + cut = keys[ TextCompletion ]; + + //using just the standard Ctrl+E isn't user-friendly enough for Grandma... + if ( completing && ( cut.contains( key ) || e->key() == TQt::Key_Enter || e->key() == TQt::Key_Return ) ) { + int paraFrom, indexFrom, paraTo, indexTo; + getSelection ( ¶From, &indexFrom, ¶To, &indexTo ); + + removeSelection(); + setCursorPosition( paraTo, indexTo ); + + completing = false; + return ; + } + + // handle rotation + + // Handles previous match + if ( keys[ PrevCompletionMatch ].isNull() ) + cut = TDEStdAccel::shortcut( TDEStdAccel::PrevCompletion ); + else + cut = keys[ PrevCompletionMatch ]; + + if ( cut.contains( key ) ) { + rotateText( TDECompletionBase::PrevCompletionMatch ); + return ; + } + + // Handles next match + if ( keys[ NextCompletionMatch ].isNull() ) + cut = TDEStdAccel::shortcut( TDEStdAccel::NextCompletion ); + else + cut = keys[ NextCompletionMatch ]; + + if ( cut.contains( key ) ) { + rotateText( TDECompletionBase::NextCompletionMatch ); + return ; + } + + //any other key events will end any text completion execpt for modifiers + switch ( e->key() ) { + case TQt::Key_Shift: + case TQt::Key_Control: + case TQt::Key_Alt: + case TQt::Key_Meta: + break; + default: + completing = false; + break; + } + + // Let KTextEdit handle any other keys events. + KTextEdit::keyPressEvent ( e ); +} + +void KreTextEdit::setCompletedText( const TQString &txt ) +{ + int para, index; + getCursorPosition( ¶, &index ); + + TQString para_text = text( para ); + int word_length = index - completion_begin; + + insert( txt.right( txt.length() - word_length ) ); + setSelection( para, index, para, completion_begin + txt.length() ); + setCursorPosition( para, index ); + + completing = true; +} + +void KreTextEdit::setCompletedItems( const TQStringList &/*items*/ ) +{} + +void KreTextEdit::tryCompletion() +{ + int para, index; + getCursorPosition( ¶, &index ); + + TQString para_text = text( para ); + if ( para_text.at( index ).isSpace() || completing ) { + if ( !completing ) + completion_begin = para_text.findRev( ' ', index - 1 ) + 1; + + TQString completing_word = para_text.mid( completion_begin, index - completion_begin ); + + TQString match = compObj() ->makeCompletion( completing_word ); + + if ( !match.isNull() && match != completing_word ) + setCompletedText( match ); + else + completing = false; + } +} + +void KreTextEdit::rotateText( TDECompletionBase::KeyBindingType type ) +{ + TDECompletion * comp = compObj(); + if ( comp && completing && + ( type == TDECompletionBase::PrevCompletionMatch || + type == TDECompletionBase::NextCompletionMatch ) ) { + TQString input = ( type == TDECompletionBase::PrevCompletionMatch ) ? comp->previousMatch() : comp->nextMatch(); + + // Skip rotation if previous/next match is null or the same text + int para, index; + getCursorPosition( ¶, &index ); + TQString para_text = text( para ); + TQString complete_word = para_text.mid( completion_begin, index - completion_begin ); + if ( input.isNull() || input == complete_word ) + return ; + setCompletedText( input ); + } +} + +void KreTextEdit::addCompletionItem( const TQString &name ) +{ + compObj() ->addItem( name ); +} + +void KreTextEdit::removeCompletionItem( const TQString &name ) +{ + compObj() ->removeItem( name ); +} + +void KreTextEdit::clearCompletionItems() +{ + compObj() ->clear(); +} + +#include "kretextedit.moc" diff --git a/src/widgets/kretextedit.h b/src/widgets/kretextedit.h new file mode 100644 index 0000000..ec9e5fa --- /dev/null +++ b/src/widgets/kretextedit.h @@ -0,0 +1,52 @@ +/*************************************************************************** +* Copyright (C) 2004 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef KRETEXTEDIT_H +#define KRETEXTEDIT_H + +#include <ktextedit.h> +#include <kcompletion.h> + +#include "datablocks/elementlist.h" + +/* @author Jason Kivlighn + * @brief An extended KTextEdit that allows for text completion + */ +class KreTextEdit : public KTextEdit, TDECompletionBase +{ + TQ_OBJECT + +public: + KreTextEdit( TQWidget *parent ); + + virtual void setCompletedText( const TQString &text ); + virtual void setCompletedItems( const TQStringList &items ); + +public slots: + void addCompletionItem( const TQString & ); + void removeCompletionItem( const TQString & ); + void clearCompletionItems(); + +protected: + void keyPressEvent( TQKeyEvent * ); + +private slots: + void haltCompletion(); + +private: + void tryCompletion(); + void rotateText( TDECompletionBase::KeyBindingType type ); + + bool completing; + int completion_begin; + +}; + +#endif //KRETEXTEDIT_H diff --git a/src/widgets/kwidgetlistbox.cpp b/src/widgets/kwidgetlistbox.cpp new file mode 100644 index 0000000..a05ddcb --- /dev/null +++ b/src/widgets/kwidgetlistbox.cpp @@ -0,0 +1,210 @@ +/* + * Copyright (C) 2005 Petri Damst�n <[email protected]> + * + * Note: This file is now part of Krecipes, which is a slightly modified version of the + * original used in SuperKaramba + * + * This file is part of SuperKaramba. + * + * SuperKaramba 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. + * + * SuperKaramba 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 SuperKaramba; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + ****************************************************************************/ +#include "kwidgetlistbox.h" +#include <kdebug.h> +#include <tdeglobalsettings.h> + +KWidgetListbox::KWidgetListbox(TQWidget *parent, const char *name) + : TQTable(parent, name) +{ + setNumRows(0); + setNumCols(1); + setColumnStretchable(0, true); + setLeftMargin(0); + setTopMargin(0); + horizontalHeader()->hide(); + verticalHeader()->hide(); + setSelectionMode(TQTable::NoSelection); + setFocusStyle(TQTable::FollowStyle); + connect(this, TQ_SIGNAL(currentChanged(int, int)), + this, TQ_SLOT(selectionChanged(int, int))); + setHScrollBarMode(TQScrollView::AlwaysOff); + setVScrollBarMode(TQScrollView::Auto); +} + +KWidgetListbox::~KWidgetListbox() +{ + clear(); +} + +void KWidgetListbox::clear() +{ + for(int i = 0; i < numRows(); ++i) + clearCellWidget(i, 0); + setNumRows(0); +} + +int KWidgetListbox::insertItem(TQWidget* item, int index) +{ + int row = index; + + if(index == -1) + { + row = numRows(); + } + //else + // return -1; + + insertRows(row); + setRowHeight(row, item->height()); + setCellWidget(row, 0, item); + + for ( int i = row; i < numRows(); ++i ) { + setItemColors(i, even(i)); + } + + ensureCellVisible(row,0); + + return row; +} + +void KWidgetListbox::setSelected(TQWidget* item) +{ + setSelected(index(item)); +} + +void KWidgetListbox::selectionChanged(int row, int col) +{ + ensureCellVisible(row, col); + updateColors(); + emit selected(row); +} + +void KWidgetListbox::removeItem(TQWidget* item) +{ + removeItem(index(item)); +} + +void KWidgetListbox::removeItem(int index) +{ + removeRow(index); + updateColors(); +} + +void KWidgetListbox::setSelected(int index) +{ + setCurrentCell(index, 0); +} + +int KWidgetListbox::selected() const +{ + return currentRow(); +} + +TQWidget* KWidgetListbox::selectedItem() const +{ + return item(selected()); +} + +TQWidget* KWidgetListbox::item(int index) const +{ + return cellWidget(index, 0); +} + +int KWidgetListbox::index(TQWidget* itm) const +{ + for(int i = 0; i < numRows(); ++i) + if(item(i) == itm) + return i; + return -1; +} + +bool KWidgetListbox::even(int index) +{ + int v = 0; + for(int i = 0; i < numRows(); ++i) + { + if(index == i) + break; + //if(!isRowHidden(i)) + ++v; + } + return (v%2 == 0); +} + +void KWidgetListbox::updateColors() +{ + int v = 0; + for(int i = 0; i < numRows(); ++i) + { + //if(!isRowHidden(i)) + { + setItemColors(i, (v%2 == 0)); + ++v; + } + } +} + +void KWidgetListbox::setItemColors(int index, bool even) +{ + TQWidget* itm = item(index); +if ( !itm){ kdDebug()<<"no widget at index "<<index<<endl; return; } +/* + if(index == selected()) + { + itm->setPaletteBackgroundColor(TDEGlobalSettings::highlightColor()); + itm->setPaletteForegroundColor(TDEGlobalSettings::highlightedTextColor()); + }*/ + if(even) + { + itm->setPaletteBackgroundColor(TDEGlobalSettings::baseColor()); + itm->setPaletteForegroundColor(TDEGlobalSettings::textColor()); + } + else + { + itm->setPaletteBackgroundColor( + TDEGlobalSettings::alternateBackgroundColor()); + itm->setPaletteForegroundColor(TDEGlobalSettings::textColor()); + } +} + +void KWidgetListbox::showItems(show_callback func, void* data) +{ + for(int i = 0; i < numRows(); ++i) + { + if(func == 0) + showRow(i); + else + { + if(func(i, item(i), data)) + showRow(i); + else + hideRow(i); + } + } + updateColors(); +} + +void KWidgetListbox::showEvent(TQShowEvent*) +{ + //kdDebug() << k_funcinfo << endl; + repaintContents(false); +} + +void KWidgetListbox::paintCell(TQPainter*, int, int, const TQRect&, + bool, const TQColorGroup&) +{ + //kdDebug() << k_funcinfo << endl; +} + +#include "kwidgetlistbox.moc" diff --git a/src/widgets/kwidgetlistbox.h b/src/widgets/kwidgetlistbox.h new file mode 100644 index 0000000..0ad463c --- /dev/null +++ b/src/widgets/kwidgetlistbox.h @@ -0,0 +1,71 @@ +/* + * Copyright (C) 2005 Petri Damst�n <[email protected]> + * + * Note: This file is now part of Krecipes, which is a slightly modified version of the + * original used in SuperKaramba + * + * This file is part of SuperKaramba. + * + * SuperKaramba 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. + * + * SuperKaramba 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 SuperKaramba; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + ****************************************************************************/ +#ifndef KWIDGETLISTBOX_H +#define KWIDGETLISTBOX_H + +#include <tqtable.h> + +/** +@author See README for the list of authors +*/ + +typedef bool (*show_callback) (int index, TQWidget* widget, void* data); + +class KWidgetListbox : public TQTable +{ + TQ_OBJECT + + public: + KWidgetListbox(TQWidget *parent = 0, const char *name = 0); + ~KWidgetListbox(); + + int insertItem(TQWidget* item, int index = -1); + void setSelected(TQWidget* item); + void setSelected(int index); + void removeItem(TQWidget* item); + void removeItem(int index); + void clear(); + int selected() const; + TQWidget* selectedItem() const; + TQWidget* item(int index) const; + int index(TQWidget* itm) const; + uint count() const { return numRows(); }; + + void showItems(show_callback func = 0, void* data = 0); + + void paintCell(TQPainter* p, int row, int col, const TQRect& cr, + bool selected, const TQColorGroup& cg); + protected: + void setItemColors(int index, bool even); + void updateColors(); + bool even(int index); + virtual void showEvent(TQShowEvent* e); + + protected slots: + void selectionChanged(int row, int col); + + signals: + void selected(int index); +}; + +#endif diff --git a/src/widgets/paneldeco.cpp b/src/widgets/paneldeco.cpp new file mode 100644 index 0000000..8045a5b --- /dev/null +++ b/src/widgets/paneldeco.cpp @@ -0,0 +1,173 @@ +/*************************************************************************** +* Copyright (C) 2003 by Unai Garro ([email protected]) * +* * +* * +* 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. * +***************************************************************************/ +#include "paneldeco.h" + +#include <tqpainter.h> +#include <tqpoint.h> +#include <tqrect.h> + +#include <kiconloader.h> +#include <kpixmap.h> +#include <kpixmapeffect.h> + + +// Panel decoration + +PanelDeco::PanelDeco( TQWidget *parent, const char *name, const TQString &title, const TQString &iconName ) : TQVBox( parent, name ) +{ + + // Top decoration + tDeco = new TopDeco( this, "TopDecoration", title, iconName ); + + hbox = new TQHBox( this ); + + //Left decoration + lDeco = new LeftDeco( hbox, "LeftDecoration" ); + + //The widget stack (panels) + stack = new TQWidgetStack( hbox ); + stack->setSizePolicy( TQSizePolicy( TQSizePolicy::MinimumExpanding, TQSizePolicy::MinimumExpanding ) ); + +} + + +PanelDeco::~PanelDeco() +{} + +void PanelDeco::childEvent( TQChildEvent *e ) +{ + if ( e->type() == TQEvent::ChildInserted ) { + TQObject * obj = e->child(); + if ( obj->inherits( "TQWidget" ) ) { + TQWidget * w = ( TQWidget* ) obj; + if ( w != hbox && w != tDeco ) + w->reparent( stack, TQPoint( 0, 0 ) ); + } + } +} + + +int PanelDeco::id( TQWidget* w ) +{ + return ( stack->id( w ) ); +} + +void PanelDeco::raise( TQWidget *w ) +{ + TQWidget * old_w = visiblePanel(); + + stack->raiseWidget( w ); + + if ( old_w != w ) + emit panelRaised( w, old_w ); +} + +TQWidget* PanelDeco::visiblePanel( void ) +{ + return ( stack->visibleWidget() ); +} + +void PanelDeco::setHeader( const TQString &title, const TQString &icon ) +{ + tDeco->setHeader( title, icon ); +} + +// Left part of the decoration + +LeftDeco::LeftDeco( TQWidget *parent, const char *name ) : + TQWidget( parent, name, TQt::WNoAutoErase ) +{} + +LeftDeco::~LeftDeco() +{} + +// Top part of the decoration + +TopDeco::TopDeco( TQWidget *parent, const char *name, const TQString &title, const TQString &iconName ) : + TQWidget( parent, name, TQt::WNoAutoErase ) +{ + setMinimumHeight( 30 ); + icon = 0; + panelTitle = TQString::null; + if ( !iconName.isNull() ) { + TDEIconLoader il; + icon = new TQPixmap( il.loadIcon( iconName, TDEIcon::NoGroup, 22 ) ); + } + + if ( !title.isNull() ) { + panelTitle = title; + } +} + +TopDeco::~TopDeco() +{ + delete icon; +} + + +void TopDeco::paintEvent( TQPaintEvent * ) +{ + // Get gradient colors + TQColor c1 = colorGroup().button().light( 120 ); + TQColor c2 = paletteBackgroundColor(); + + // Draw the gradient + KPixmap kpm; + kpm.resize( size() ); + KPixmapEffect::unbalancedGradient ( kpm, c1, c2, KPixmapEffect::VerticalGradient, 150, 150 ); + + // Add a line on top + TQPainter painter( &kpm ); + painter.setPen( colorGroup().button().dark( 130 ) ); + painter.drawLine( 0, 0, width(), 0 ); + + // Now Add the icon + int xpos = 0, ypos = 0; + if ( icon ) { + xpos = 20; + ypos = ( height() - icon->height() ) / 2 - 1; + painter.drawPixmap( xpos, ypos, *icon ); + xpos += icon->width(); // Move it so that later we can easily place the text + } + + // Finally, draw the text besides the icon + if ( !panelTitle.isNull() ) { + xpos += 15; + TQRect r = rect(); + r.setLeft( xpos ); + painter.setPen( TQColor( 0x00, 0x00, 0x00 ) ); + TQFont ft = font(); + ft.setBold( true ); + painter.setFont( ft ); + painter.drawText( r, TQt::AlignVCenter, panelTitle ); + } + painter.end(); + // Copy the pixmap to the widget + bitBlt( this, 0, 0, &kpm ); +} + +void TopDeco::setHeader( const TQString &title, const TQString &iconName ) +{ + if ( !title.isNull() ) + panelTitle = title; + if ( !iconName.isNull() ) { + TDEIconLoader il; + icon = new TQPixmap( il.loadIcon( iconName, TDEIcon::NoGroup, 22 ) ); + } + if ( !title.isNull() || !iconName.isNull() ) + update(); +} + +TQSize TopDeco::sizeHint( void ) +{ + return ( TQSize( parentWidget() ->width(), 30 ) ); +} + +#include "paneldeco.moc" diff --git a/src/widgets/paneldeco.h b/src/widgets/paneldeco.h new file mode 100644 index 0000000..126f7fb --- /dev/null +++ b/src/widgets/paneldeco.h @@ -0,0 +1,85 @@ +/*************************************************************************** +* Copyright (C) 2003 by Unai Garro ([email protected]) * +* * +* * +* 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. * +***************************************************************************/ +#ifndef PANELDECO_H +#define PANELDECO_H + + +#include <tqevent.h> +#include <tqiconset.h> +#include <tqstring.h> +#include <tqhbox.h> +#include <tqpixmap.h> +#include <tqvbox.h> +#include <tqwidget.h> +#include <tqwidgetstack.h> + + +/** +* @author Unai Garro +*/ + +class PanelDeco; +class LeftDeco; +class TopDeco; + +class PanelDeco : public TQVBox +{ + TQ_OBJECT +public: + // Methods + PanelDeco( TQWidget *parent = 0, const char *name = 0, const TQString &title = TQString::null, const TQString &iconName = TQString::null ); + ~PanelDeco(); + int id( TQWidget* w ); // obtain the id of the given panel + TQWidget* visiblePanel( void ); // obtain the current active panel no. + +signals: + void panelRaised( TQWidget *w, TQWidget *old_w ); + +private: + TQHBox *hbox; + LeftDeco *lDeco; + TopDeco *tDeco; + TQWidgetStack *stack; + +public slots: + void raise( TQWidget *w ); + void setHeader( const TQString &title = TQString::null, const TQString &icon = TQString::null ); +protected: + virtual void childEvent( TQChildEvent *e ); + + +}; + +class LeftDeco: public TQWidget +{ + TQ_OBJECT +public: + LeftDeco( TQWidget *parent = 0, const char *name = 0 ); + + ~LeftDeco(); +}; + +class TopDeco: public TQWidget +{ + TQ_OBJECT +public: + TopDeco( TQWidget *parent = 0, const char *name = 0, const TQString &title = TQString::null, const TQString &iconName = TQString::null ); + ~TopDeco(); + virtual TQSize sizeHint( void ); +public slots: + void setHeader( const TQString &title = TQString::null, const TQString &iconName = TQString::null ); +protected: + virtual void paintEvent( TQPaintEvent *e ); +private: + TQPixmap *icon; + TQString panelTitle; +}; + +#endif diff --git a/src/widgets/prepmethodcombobox.cpp b/src/widgets/prepmethodcombobox.cpp new file mode 100644 index 0000000..b0b7660 --- /dev/null +++ b/src/widgets/prepmethodcombobox.cpp @@ -0,0 +1,186 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "prepmethodcombobox.h" + +#include <tqlistbox.h> + +#include <kdebug.h> + +#include "backends/recipedb.h" +#include "datablocks/elementlist.h" + +/** Completion object which allows completing completing items + * the last item in a comma-separated list + */ +class PrepMethodCompletion : public TDECompletion +{ +public: + PrepMethodCompletion() : TDECompletion() + {} + + virtual TQString makeCompletion( const TQString &string ) { + kdDebug()<<"original makeCompletion( "<<string<<" )"<<endl; + + int comma_index = string.findRev(","); + TQString completion_txt = string; + if ( comma_index != -1 ) + completion_txt = completion_txt.right( completion_txt.length() - comma_index - 1 ).stripWhiteSpace(); + if ( completion_txt.isEmpty() ) + return string; + + kdDebug()<<"altered makeCompletion( "<<completion_txt<<" )"<<endl; + + completion_txt = TDECompletion::makeCompletion(completion_txt); + kdDebug()<<"got: "<<completion_txt<<endl; + + if ( completion_txt.isEmpty() ) + completion_txt = string; + else if ( comma_index != -1 ) + completion_txt = string.left( comma_index ) + "," + completion_txt; + + kdDebug()<<"returning: "<<completion_txt<<endl; + return completion_txt; + } +}; + +PrepMethodComboBox::PrepMethodComboBox( bool b, TQWidget *parent, RecipeDB *db, const TQString &specialItem ) : + KComboBox( b, parent ), + database( db ), m_specialItem(specialItem) +{ + setAutoDeleteCompletionObject(true); + setCompletionObject(new PrepMethodCompletion()); +} + +void PrepMethodComboBox::reload() +{ + TQString remember_text; + if ( editable() ) + remember_text = lineEdit()->text(); + + ElementList prepMethodList; + database->loadPrepMethods( &prepMethodList ); + + clear(); + prepMethodComboRows.clear(); + + int row = 0; + if ( !m_specialItem.isNull() ) { + insertItem(m_specialItem); + prepMethodComboRows.insert( row, -1 ); + row++; + } + for ( ElementList::const_iterator it = prepMethodList.begin(); it != prepMethodList.end(); ++it, ++row ) { + insertItem((*it).name); + completionObject()->addItem((*it).name); + prepMethodComboRows.insert( row,(*it).id ); + } + + if ( editable() ) + lineEdit()->setText( remember_text ); + + database->disconnect( this ); + connect( database, TQ_SIGNAL( prepMethodCreated( const Element & ) ), TQ_SLOT( createPrepMethod( const Element & ) ) ); + connect( database, TQ_SIGNAL( prepMethodRemoved( int ) ), TQ_SLOT( removePrepMethod( int ) ) ); +} + +int PrepMethodComboBox::id( int row ) +{ + return prepMethodComboRows[ row ]; +} + +int PrepMethodComboBox::id( const TQString &ing ) +{ + for ( int i = 0; i < count(); i++ ) { + if ( ing == text( i ) ) + return id(i); + } + kdDebug()<<"Warning: couldn't find the ID for "<<ing<<endl; + return -1; +} + +void PrepMethodComboBox::createPrepMethod( const Element &element ) +{ + int row = findInsertionPoint( element.name ); + + TQString remember_text; + if ( editable() ) + remember_text = lineEdit()->text(); + + insertItem( element.name, row ); + completionObject()->addItem(element.name); + + if ( editable() ) + lineEdit()->setText( remember_text ); + + //now update the map by pushing everything after this item down + TQMap<int, int> new_map; + for ( TQMap<int, int>::iterator it = prepMethodComboRows.begin(); it != prepMethodComboRows.end(); ++it ) { + if ( it.key() >= row ) { + new_map.insert( it.key() + 1, it.data() ); + } + else + new_map.insert( it.key(), it.data() ); + } + prepMethodComboRows = new_map; + prepMethodComboRows.insert( row, element.id ); +} + +void PrepMethodComboBox::removePrepMethod( int id ) +{ + int row = -1; + for ( TQMap<int, int>::iterator it = prepMethodComboRows.begin(); it != prepMethodComboRows.end(); ++it ) { + if ( it.data() == id ) { + row = it.key(); + completionObject()->removeItem( text(row) ); + removeItem( row ); + prepMethodComboRows.remove( it ); + break; + } + } + + if ( row == -1 ) + return ; + + //now update the map by pushing everything after this item up + TQMap<int, int> new_map; + for ( TQMap<int, int>::iterator it = prepMethodComboRows.begin(); it != prepMethodComboRows.end(); ++it ) { + if ( it.key() > row ) { + new_map.insert( it.key() - 1, it.data() ); + } + else + new_map.insert( it.key(), it.data() ); + } + prepMethodComboRows = new_map; +} + +int PrepMethodComboBox::findInsertionPoint( const TQString &name ) +{ + for ( int i = 0; i < count(); i++ ) { + if ( TQString::localeAwareCompare( name, text( i ) ) < 0 ) + return i; + } + + return count(); +} + +void PrepMethodComboBox::setSelected( int prepID ) +{ + //do a reverse lookup on the row->id map + TQMap<int, int>::const_iterator it; + for ( it = prepMethodComboRows.begin(); it != prepMethodComboRows.end(); ++it ) { + if ( it.data() == prepID ) { + setCurrentItem(it.key()); + break; + } + } +} + +#include "prepmethodcombobox.moc" diff --git a/src/widgets/prepmethodcombobox.h b/src/widgets/prepmethodcombobox.h new file mode 100644 index 0000000..ad902d7 --- /dev/null +++ b/src/widgets/prepmethodcombobox.h @@ -0,0 +1,48 @@ +/*************************************************************************** +* Copyright (C) 2005 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef PREPMETHODCOMBOBOX_H +#define PREPMETHODCOMBOBOX_H + +#include <kcombobox.h> + +#include <tqmap.h> + +#include "datablocks/element.h" + +class RecipeDB; +class ElementList; + +class PrepMethodComboBox : public KComboBox +{ + TQ_OBJECT + +public: + PrepMethodComboBox( bool, TQWidget *parent, RecipeDB *db, const TQString &specialItem = TQString::null ); + + void reload(); + int id( int row ); + int id( const TQString &ing ); + void setSelected( int prepID ); + +private slots: + void createPrepMethod( const Element &element ); + void removePrepMethod( int id ); + + int findInsertionPoint( const TQString &name ); + +private: + RecipeDB *database; + TQMap<int, int> prepMethodComboRows; // Contains the prep method id for every given row in the combobox + TQString m_specialItem; +}; + +#endif //PREPMETHODCOMBOBOX_H + diff --git a/src/widgets/prepmethodlistview.cpp b/src/widgets/prepmethodlistview.cpp new file mode 100644 index 0000000..1f1939b --- /dev/null +++ b/src/widgets/prepmethodlistview.cpp @@ -0,0 +1,189 @@ +/*************************************************************************** +* Copyright (C) 2004 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "prepmethodlistview.h" + +#include <tdemessagebox.h> +#include <tdeconfig.h> +#include <tdelocale.h> +#include <tdeglobal.h> +#include <kiconloader.h> +#include <tdepopupmenu.h> + +#include "backends/recipedb.h" +#include "dialogs/createelementdialog.h" +#include "dialogs/dependanciesdialog.h" + +PrepMethodListView::PrepMethodListView( TQWidget *parent, RecipeDB *db ) : DBListViewBase( parent,db,db->prepMethodCount()) +{ + setAllColumnsShowFocus( true ); + setDefaultRenameAction( TQListView::Reject ); +} + +void PrepMethodListView::init() +{ + connect( database, TQ_SIGNAL( prepMethodCreated( const Element & ) ), TQ_SLOT( checkCreatePrepMethod( const Element & ) ) ); + connect( database, TQ_SIGNAL( prepMethodRemoved( int ) ), TQ_SLOT( removePrepMethod( int ) ) ); +} + +void PrepMethodListView::load( int limit, int offset ) +{ + ElementList prepMethodList; + database->loadPrepMethods( &prepMethodList, limit, offset ); + + setTotalItems(prepMethodList.count()); + + for ( ElementList::const_iterator ing_it = prepMethodList.begin(); ing_it != prepMethodList.end(); ++ing_it ) + createPrepMethod( *ing_it ); +} + +void PrepMethodListView::checkCreatePrepMethod( const Element &el ) +{ + if ( handleElement(el.name) ) { //only create this prep method if the base class okays it + createPrepMethod(el); + } +} + + +StdPrepMethodListView::StdPrepMethodListView( TQWidget *parent, RecipeDB *db, bool editable ) : PrepMethodListView( parent, db ) +{ + addColumn( i18n( "Preparation Method" ) ); + + TDEConfig * config = TDEGlobal::config(); + config->setGroup( "Advanced" ); + bool show_id = config->readBoolEntry( "ShowID", false ); + addColumn( i18n( "Id" ), show_id ? -1 : 0 ); + + if ( editable ) { + setRenameable( 0, true ); + + TDEIconLoader *il = new TDEIconLoader; + + kpop = new TDEPopupMenu( this ); + kpop->insertItem( il->loadIcon( "document-new", TDEIcon::NoGroup, 16 ), i18n( "&Create" ), this, TQ_SLOT( createNew() ), CTRL + Key_C ); + kpop->insertItem( il->loadIcon( "edit-delete", TDEIcon::NoGroup, 16 ), i18n( "&Delete" ), this, TQ_SLOT( remove + () ), Key_Delete ); + kpop->insertItem( il->loadIcon( "edit", TDEIcon::NoGroup, 16 ), i18n( "&Rename" ), this, TQ_SLOT( rename() ), CTRL + Key_R ); + kpop->polish(); + + delete il; + + connect( this, TQ_SIGNAL( contextMenu( TDEListView *, TQListViewItem *, const TQPoint & ) ), TQ_SLOT( showPopup( TDEListView *, TQListViewItem *, const TQPoint & ) ) ); + connect( this, TQ_SIGNAL( doubleClicked( TQListViewItem* ) ), this, TQ_SLOT( modPrepMethod( TQListViewItem* ) ) ); + connect( this, TQ_SIGNAL( itemRenamed( TQListViewItem* ) ), this, TQ_SLOT( savePrepMethod( TQListViewItem* ) ) ); + } +} + +void StdPrepMethodListView::showPopup( TDEListView * /*l*/, TQListViewItem *i, const TQPoint &p ) +{ + if ( i ) + kpop->exec( p ); +} + +void StdPrepMethodListView::createNew() +{ + CreateElementDialog * elementDialog = new CreateElementDialog( this, i18n( "New Preparation Method" ) ); + + if ( elementDialog->exec() == TQDialog::Accepted ) { + TQString result = elementDialog->newElementName(); + + //check bounds first + if ( checkBounds( result ) ) + database->createNewPrepMethod( result ); // Create the new prepMethod in the database + } +} + +void StdPrepMethodListView::remove + () +{ + TQListViewItem * item = currentItem(); + + if ( item ) { + ElementList dependingRecipes; + int prepMethodID = item->text( 1 ).toInt(); + database->findPrepMethodDependancies( prepMethodID, &dependingRecipes ); + if ( dependingRecipes.isEmpty() ) + database->removePrepMethod( prepMethodID ); + else // Need Warning! + { + ListInfo info; + info.list = dependingRecipes; + info.name = i18n("Recipes"); + DependanciesDialog warnDialog( this, info ); + warnDialog.setCustomWarning( i18n("You are about to permanantly delete recipes from your database.") ); + if ( warnDialog.exec() == TQDialog::Accepted ) + database->removePrepMethod( prepMethodID ); + } + } +} + +void StdPrepMethodListView::rename() +{ + TQListViewItem * item = currentItem(); + + if ( item ) + PrepMethodListView::rename( item, 0 ); +} + +void StdPrepMethodListView::createPrepMethod( const Element &ing ) +{ + createElement(new TQListViewItem( this, ing.name, TQString::number( ing.id ) )); +} + +void StdPrepMethodListView::removePrepMethod( int id ) +{ + TQListViewItem * item = findItem( TQString::number( id ), 1 ); + removeElement(item); +} + +void StdPrepMethodListView::modPrepMethod( TQListViewItem* i ) +{ + if ( i ) + PrepMethodListView::rename( i, 0 ); +} + +void StdPrepMethodListView::savePrepMethod( TQListViewItem* i ) +{ + if ( !checkBounds( i->text( 0 ) ) ) { + reload(ForceReload); //reset the changed text + return ; + } + + int existing_id = database->findExistingPrepByName( i->text( 0 ) ); + int prep_id = i->text( 1 ).toInt(); + if ( existing_id != -1 && existing_id != prep_id ) //category already exists with this label... merge the two + { + switch ( KMessageBox::warningContinueCancel( this, i18n( "This preparation method already exists. Continuing will merge these two into one. Are you sure?" ) ) ) + { + case KMessageBox::Continue: { + database->mergePrepMethods( existing_id, prep_id ); + break; + } + default: + reload(ForceReload); + break; + } + } + else { + database->modPrepMethod( ( i->text( 1 ) ).toInt(), i->text( 0 ) ); + } +} + +bool StdPrepMethodListView::checkBounds( const TQString &name ) +{ + if ( name.length() > uint(database->maxPrepMethodNameLength()) ) { + KMessageBox::error( this, TQString( i18n( "Preparation method cannot be longer than %1 characters." ) ).arg( database->maxPrepMethodNameLength() ) ); + return false; + } + + return true; +} + +#include "prepmethodlistview.moc" diff --git a/src/widgets/prepmethodlistview.h b/src/widgets/prepmethodlistview.h new file mode 100644 index 0000000..a31fccd --- /dev/null +++ b/src/widgets/prepmethodlistview.h @@ -0,0 +1,70 @@ +/*************************************************************************** +* Copyright (C) 2004 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef PREPMETHODLISTVIEW_H +#define PREPMETHODLISTVIEW_H + +#include "dblistviewbase.h" + +#include "datablocks/element.h" + +class RecipeDB; +class TDEPopupMenu; + +class PrepMethodListView : public DBListViewBase +{ + TQ_OBJECT + +public: + PrepMethodListView( TQWidget *parent, RecipeDB *db ); + +public slots: + virtual void load( int curr_limit, int curr_offset ); + +protected slots: + virtual void createPrepMethod( const Element & ) = 0; + virtual void removePrepMethod( int ) = 0; + + void checkCreatePrepMethod( const Element &el ); + +protected: + virtual void init(); +}; + + +class StdPrepMethodListView : public PrepMethodListView +{ + TQ_OBJECT + +public: + StdPrepMethodListView( TQWidget *parent, RecipeDB *db, bool editable = false ); + +protected: + virtual void createPrepMethod( const Element & ); + virtual void removePrepMethod( int ); + +private slots: + void showPopup( TDEListView *, TQListViewItem *, const TQPoint & ); + + void createNew(); + void remove + (); + void rename(); + + void modPrepMethod( TQListViewItem* i ); + void savePrepMethod( TQListViewItem* i ); + +private: + bool checkBounds( const TQString &name ); + + TDEPopupMenu *kpop; +}; + +#endif //PREPMETHODLISTVIEW_H diff --git a/src/widgets/propertylistview.cpp b/src/widgets/propertylistview.cpp new file mode 100644 index 0000000..29a3321 --- /dev/null +++ b/src/widgets/propertylistview.cpp @@ -0,0 +1,289 @@ +/*************************************************************************** +* Copyright (C) 2004 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "propertylistview.h" + +#include <tdelocale.h> +#include <tdemessagebox.h> +#include <tdeconfig.h> +#include <tdeglobal.h> +#include <kiconloader.h> +#include <tdepopupmenu.h> +#include <kdebug.h> + +#include "backends/recipedb.h" +#include "dialogs/createpropertydialog.h" + +PropertyCheckListItem::PropertyCheckListItem( TQListView* klv, const IngredientProperty &property ) : TQCheckListItem( klv, TQString::null, TQCheckListItem::CheckBox ), + m_property( property ) +{ + //setOn( false ); // Set unchecked by default +} + +PropertyCheckListItem::PropertyCheckListItem( TQListViewItem* it, const IngredientProperty &property ) : TQCheckListItem( it, TQString::null, TQCheckListItem::CheckBox ), + m_property( property ) +{ + //setOn( false ); // Set unchecked by default +} + +TQString PropertyCheckListItem::text( int column ) const +{ + switch ( column ) { + case 0: + return m_property.name; + break; + case 1: + return m_property.units; + break; + case 2: + return TQString::number( m_property.id ); + break; + + } + + return TQString::null; +} + + +HidePropertyCheckListItem::HidePropertyCheckListItem( TQListView* klv, const IngredientProperty &property, bool enable ) : PropertyCheckListItem( klv, property ) +{ + m_holdSettings = true; + setOn( enable ); // Set checked by default + m_holdSettings = false; +} + +HidePropertyCheckListItem::HidePropertyCheckListItem( TQListViewItem* it, const IngredientProperty &property, bool enable ) : PropertyCheckListItem( it, property ) +{ + m_holdSettings = true; + setOn( enable ); // Set checked by default + m_holdSettings = false; +} + +void HidePropertyCheckListItem::stateChange( bool on ) +{ + if ( !m_holdSettings ) { + TDEConfig *config = TDEGlobal::config(); + config->setGroup("Formatting"); + + config->sync(); + TQStringList hiddenList = config->readListEntry("HiddenProperties"); + if ( on ) + hiddenList.remove(m_property.name); + else if ( !hiddenList.contains(m_property.name) ) + hiddenList.append(m_property.name); + + config->writeEntry("HiddenProperties",hiddenList); + } +} + +PropertyListView::PropertyListView( TQWidget *parent, RecipeDB *db ) : TDEListView( parent ), + database( db ) +{ + setAllColumnsShowFocus( true ); + setDefaultRenameAction( TQListView::Reject ); + + connect( db, TQ_SIGNAL( propertyCreated( const IngredientProperty & ) ), TQ_SLOT( createProperty( const IngredientProperty & ) ) ); + connect( db, TQ_SIGNAL( propertyRemoved( int ) ), TQ_SLOT( removeProperty( int ) ) ); +} + +void PropertyListView::reload() +{ + clear(); // Clear the view + + m_loading = true; + + IngredientPropertyList propertyList; + database->loadProperties( &propertyList ); + + //Populate this data into the TDEListView + IngredientPropertyList::const_iterator prop_it; + for ( prop_it = propertyList.begin(); prop_it != propertyList.end(); ++prop_it ) + createProperty( *prop_it ); + + m_loading = false; +} + + + +StdPropertyListView::StdPropertyListView( TQWidget *parent, RecipeDB *db, bool editable ) : PropertyListView( parent, db ) +{ + addColumn( i18n( "Property" ) ); + addColumn( i18n( "Units" ) ); + + TDEConfig * config = TDEGlobal::config(); + config->setGroup( "Advanced" ); + bool show_id = config->readBoolEntry( "ShowID", false ); + addColumn( i18n( "Id" ), show_id ? -1 : 2 ); + + setSorting( 0 ); + + if ( editable ) { + setRenameable( 0, true ); + + TDEIconLoader *il = new TDEIconLoader; + + kpop = new TDEPopupMenu( this ); + kpop->insertItem( il->loadIcon( "document-new", TDEIcon::NoGroup, 16 ), i18n( "&Create" ), this, TQ_SLOT( createNew() ), CTRL + Key_C ); + kpop->insertItem( il->loadIcon( "edit-delete", TDEIcon::NoGroup, 16 ), i18n( "&Delete" ), this, TQ_SLOT( remove + () ), Key_Delete ); + kpop->insertItem( il->loadIcon( "edit", TDEIcon::NoGroup, 16 ), i18n( "&Rename" ), this, TQ_SLOT( rename() ), CTRL + Key_R ); + kpop->polish(); + + delete il; + + connect( this, TQ_SIGNAL( contextMenu( TDEListView *, TQListViewItem *, const TQPoint & ) ), TQ_SLOT( showPopup( TDEListView *, TQListViewItem *, const TQPoint & ) ) ); + connect( this, TQ_SIGNAL( doubleClicked( TQListViewItem* ) ), this, TQ_SLOT( modProperty( TQListViewItem* ) ) ); + connect( this, TQ_SIGNAL( itemRenamed( TQListViewItem* ) ), this, TQ_SLOT( saveProperty( TQListViewItem* ) ) ); + } +} + +void StdPropertyListView::showPopup( TDEListView * /*l*/, TQListViewItem *i, const TQPoint &p ) +{ + if ( i ) + kpop->exec( p ); +} + +void StdPropertyListView::createNew() +{ + UnitList list; + database->loadUnits( &list ); + CreatePropertyDialog* propertyDialog = new CreatePropertyDialog( this, &list ); + + if ( propertyDialog->exec() == TQDialog::Accepted ) { + TQString name = propertyDialog->newPropertyName(); + TQString units = propertyDialog->newUnitsName(); + if ( !( ( name.isEmpty() ) || ( units.isEmpty() ) ) ) // Make sure none of the fields are empty + { + //check bounds first + if ( checkBounds( name ) ) + database->addProperty( name, units ); + } + } + delete propertyDialog; +} + +void StdPropertyListView::remove + () +{ + TQListViewItem * item = currentItem(); + + if ( item ) { + switch ( KMessageBox::warningContinueCancel( this, i18n( "Are you sure you want to delete this property?" ) ) ) { + case KMessageBox::Continue: + database->removeProperty( item->text( 2 ).toInt() ); + break; + default: + break; + } + } +} + +void StdPropertyListView::rename() +{ + TQListViewItem * item = currentItem(); + + if ( item ) + PropertyListView::rename( item, 0 ); +} + +void StdPropertyListView::removeProperty( int id ) +{ + TQListViewItem * item = findItem( TQString::number( id ), 2 ); + + Q_ASSERT( item ); + + delete item; +} + +void StdPropertyListView::createProperty( const IngredientProperty &property ) +{ + ( void ) new TQListViewItem( this, property.name, property.units, TQString::number( property.id ) ); +} + +void StdPropertyListView::modProperty( TQListViewItem* i ) +{ + if ( i ) + PropertyListView::rename( i, 0 ); +} + +void StdPropertyListView::saveProperty( TQListViewItem* i ) +{ + if ( !checkBounds( i->text( 0 ) ) ) { + reload(); //reset the changed text + return ; + } +kdDebug() << "saveProp: " << i->text( 0 ) << endl; + int existing_id = database->findExistingPropertyByName( i->text( 0 ) ); + int prop_id = i->text( 2 ).toInt(); + if ( existing_id != -1 && existing_id != prop_id ) //category already exists with this label... merge the two + { + switch ( KMessageBox::warningContinueCancel( this, i18n( "This property already exists. Continuing will merge these two properties into one. Are you sure?" ) ) ) + { + case KMessageBox::Continue: { + database->mergeProperties( existing_id, prop_id ); + break; + } + default: + reload(); + break; + } + } + else + database->modProperty( prop_id, i->text( 0 ) ); +} + +bool StdPropertyListView::checkBounds( const TQString &name ) +{ + if ( name.length() > database->maxPropertyNameLength() ) { + KMessageBox::error( this, TQString( i18n( "Property name cannot be longer than %1 characters." ) ).arg( database->maxPropertyNameLength() ) ); + return false; + } + + return true; +} + + + +PropertyConstraintListView::PropertyConstraintListView( TQWidget *parent, RecipeDB *db ) : PropertyListView( parent, db ) +{ + addColumn( i18n( "Enabled" ) ); + addColumn( i18n( "Property" ) ); + addColumn( i18n( "Min. Value" ) ); + addColumn( i18n( "Max. Value" ) ); + addColumn( "Id", 0 ); //hidden, only for internal purposes + + setRenameable( 0, true ); +} + +void PropertyConstraintListView::removeProperty( int id ) +{ + TQListViewItem * item = findItem( TQString::number( id ), 4 ); + + Q_ASSERT( item ); + + delete item; +} + +void PropertyConstraintListView::createProperty( const IngredientProperty &property ) +{ + ( void ) new ConstraintsListItem( this, property ); +} + + +CheckPropertyListView::CheckPropertyListView( TQWidget *parent, RecipeDB *db, bool editable ) : StdPropertyListView( parent, db, editable ) +{ +} + +void CheckPropertyListView::createProperty( const IngredientProperty &property ) +{ + ( void ) new HidePropertyCheckListItem( this, property, (m_loading)?false:true ); +} + +#include "propertylistview.moc" diff --git a/src/widgets/propertylistview.h b/src/widgets/propertylistview.h new file mode 100644 index 0000000..62c249f --- /dev/null +++ b/src/widgets/propertylistview.h @@ -0,0 +1,203 @@ +/*************************************************************************** +* Copyright (C) 2004 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef PROPERTYLISTVIEW_H +#define PROPERTYLISTVIEW_H + +#include <tdelistview.h> + +#include "datablocks/element.h" +#include "datablocks/ingredientproperty.h" +#include "datablocks/constraintlist.h" + +class RecipeDB; +class TDEPopupMenu; + +class PropertyCheckListItem : public TQCheckListItem +{ +public: + PropertyCheckListItem( TQListView* klv, const IngredientProperty &property ); + PropertyCheckListItem( TQListViewItem* it, const IngredientProperty &property ); + + ~PropertyCheckListItem( void ) + {} + virtual TQString text( int column ) const; + + IngredientProperty property() const + { + return m_property; + } + +protected: + IngredientProperty m_property; + +}; + +class HidePropertyCheckListItem : public PropertyCheckListItem +{ +public: + HidePropertyCheckListItem( TQListView* klv, const IngredientProperty &property, bool enable = false ); + HidePropertyCheckListItem( TQListViewItem* it, const IngredientProperty &property, bool enable = false ); + +protected: + virtual void stateChange( bool on ); + +private: + bool m_holdSettings; +}; + + +class ConstraintsListItem: public TQCheckListItem +{ +public: + ConstraintsListItem( TQListView* klv, const IngredientProperty &pty ) : TQCheckListItem( klv, TQString::null, TQCheckListItem::CheckBox ) + { + // Initialize the constraint data with the the property data + ctStored = new Constraint(); + ctStored->id = pty.id; + ctStored->name = pty.name; + ctStored->perUnit = pty.perUnit; + ctStored->units = pty.units; + ctStored->max = 0; + ctStored->min = 0; + } + + ~ConstraintsListItem( void ) + { + delete ctStored; + } + +private: + Constraint *ctStored; + +public: + void setConstraint( const Constraint &constraint ) + { + delete ctStored; + ctStored = new Constraint( constraint ); + + setOn( ctStored->enabled ); + } + double maxVal() + { + return ctStored->max; + } + double minVal() + { + return ctStored->min; + } + int propertyId() + { + return ctStored->id; + } + void setMax( double maxValue ) + { + ctStored->max = maxValue; + setText( 3, TQString::number( maxValue ) ); + } + void setMin( double minValue ) + { + ctStored->min = minValue; + setText( 2, TQString::number( minValue ) ); + } + virtual TQString text( int column ) const + { + switch ( column ) { + case 1: + return ( ctStored->name ); + case 2: + return ( TQString::number( ctStored->min ) ); + case 3: + return ( TQString::number( ctStored->max ) ); + case 4: + return ( TQString::number( ctStored->id ) ); + default: + return ( TQString::null ); + } + } +}; + + +class PropertyListView : public TDEListView +{ + TQ_OBJECT + +public: + PropertyListView( TQWidget *parent, RecipeDB * ); + +public slots: + void reload( void ); + +protected: + RecipeDB *database; + bool m_loading; + +protected slots: + virtual void removeProperty( int id ) = 0; + virtual void createProperty( const IngredientProperty &property ) = 0; +}; + + + +class StdPropertyListView : public PropertyListView +{ + TQ_OBJECT + +public: + StdPropertyListView( TQWidget *parent, RecipeDB *, bool editable = false ); + +protected: + virtual void removeProperty( int id ); + virtual void createProperty( const IngredientProperty &property ); + +private slots: + void showPopup( TDEListView *, TQListViewItem *, const TQPoint & ); + + void createNew(); + void remove + (); + void rename(); + + void modProperty( TQListViewItem* i ); + void saveProperty( TQListViewItem* i ); + +private: + bool checkBounds( const TQString &name ); + + TDEPopupMenu *kpop; +}; + + + +class PropertyConstraintListView : public PropertyListView +{ +public: + PropertyConstraintListView( TQWidget *parent, RecipeDB * ); + +protected: + virtual void removeProperty( int id ); + virtual void createProperty( const IngredientProperty &property ); +}; + +class CheckPropertyListView : public StdPropertyListView +{ + TQ_OBJECT + +public: + CheckPropertyListView( TQWidget *parent, RecipeDB *, bool editable = false ); + +protected: + virtual void createProperty( const IngredientProperty &property ); + +private: + bool checkBounds( const TQString &name ); +}; + +#endif //PROPERTYLISTVIEW_H diff --git a/src/widgets/ratingdisplaywidget.ui b/src/widgets/ratingdisplaywidget.ui new file mode 100644 index 0000000..bd1380e --- /dev/null +++ b/src/widgets/ratingdisplaywidget.ui @@ -0,0 +1,232 @@ +<!DOCTYPE UI><UI version="3.1" stdsetdef="1"> +<class>RatingDisplayWidget</class> +<widget class="TQWidget"> + <property name="name"> + <cstring>RatingDisplayWidget</cstring> + </property> + <property name="geometry"> + <rect> + <x>0</x> + <y>0</y> + <width>533</width> + <height>211</height> + </rect> + </property> + <property name="paletteBackgroundColor"> + <color> + <red>250</red> + <green>248</green> + <blue>241</blue> + </color> + </property> + <hbox> + <property name="name"> + <cstring>unnamed</cstring> + </property> + <widget class="TQLayoutWidget"> + <property name="name"> + <cstring>layout4</cstring> + </property> + <vbox> + <property name="name"> + <cstring>unnamed</cstring> + </property> + <spacer> + <property name="name"> + <cstring>spacer2_2</cstring> + </property> + <property name="orientation"> + <enum>Vertical</enum> + </property> + <property name="sizeType"> + <enum>Fixed</enum> + </property> + <property name="sizeHint"> + <size> + <width>21</width> + <height>20</height> + </size> + </property> + </spacer> + <widget class="TQLabel"> + <property name="name"> + <cstring>icon</cstring> + </property> + <property name="minimumSize"> + <size> + <width>76</width> + <height>0</height> + </size> + </property> + <property name="text"> + <string></string> + </property> + <property name="alignment"> + <set>AlignCenter</set> + </property> + </widget> + <spacer> + <property name="name"> + <cstring>spacer2</cstring> + </property> + <property name="orientation"> + <enum>Vertical</enum> + </property> + <property name="sizeType"> + <enum>Expanding</enum> + </property> + <property name="sizeHint"> + <size> + <width>21</width> + <height>80</height> + </size> + </property> + </spacer> + </vbox> + </widget> + <widget class="TQLayoutWidget"> + <property name="name"> + <cstring>layout5</cstring> + </property> + <vbox> + <property name="name"> + <cstring>unnamed</cstring> + </property> + <widget class="TQLabel"> + <property name="name"> + <cstring>raterName</cstring> + </property> + <property name="sizePolicy"> + <sizepolicy> + <hsizetype>7</hsizetype> + <vsizetype>5</vsizetype> + <horstretch>0</horstretch> + <verstretch>0</verstretch> + </sizepolicy> + </property> + <property name="font"> + <font> + <bold>1</bold> + </font> + </property> + <property name="text"> + <string>Rater</string> + </property> + </widget> + <widget class="TDEListView"> + <column> + <property name="text"> + <string>Criteria</string> + </property> + <property name="clickable"> + <bool>true</bool> + </property> + <property name="resizable"> + <bool>true</bool> + </property> + </column> + <column> + <property name="text"> + <string>Stars</string> + </property> + <property name="clickable"> + <bool>true</bool> + </property> + <property name="resizable"> + <bool>true</bool> + </property> + </column> + <property name="name"> + <cstring>criteriaListView</cstring> + </property> + <property name="sizePolicy"> + <sizepolicy> + <hsizetype>5</hsizetype> + <vsizetype>5</vsizetype> + <horstretch>0</horstretch> + <verstretch>1</verstretch> + </sizepolicy> + </property> + <property name="selectionMode" stdset="0"> + <enum>NoSelection</enum> + </property> + </widget> + <widget class="TQLabel"> + <property name="name"> + <cstring>comment</cstring> + </property> + <property name="sizePolicy"> + <sizepolicy> + <hsizetype>7</hsizetype> + <vsizetype>7</vsizetype> + <horstretch>0</horstretch> + <verstretch>0</verstretch> + </sizepolicy> + </property> + <property name="text"> + <string>Comments</string> + </property> + <property name="alignment"> + <set>WordBreak|AlignTop</set> + </property> + </widget> + <widget class="TQLayoutWidget"> + <property name="name"> + <cstring>layout4</cstring> + </property> + <hbox> + <property name="name"> + <cstring>unnamed</cstring> + </property> + <spacer> + <property name="name"> + <cstring>spacer</cstring> + </property> + <property name="orientation"> + <enum>Horizontal</enum> + </property> + <property name="sizeType"> + <enum>Expanding</enum> + </property> + <property name="sizeHint"> + <size> + <width>150</width> + <height>20</height> + </size> + </property> + </spacer> + <widget class="KPushButton"> + <property name="name"> + <cstring>buttonRemove</cstring> + </property> + <property name="text"> + <string>Remove</string> + </property> + </widget> + <widget class="KPushButton"> + <property name="name"> + <cstring>buttonEdit</cstring> + </property> + <property name="text"> + <string>Edit...</string> + </property> + </widget> + </hbox> + </widget> + </vbox> + </widget> + </hbox> +</widget> +<includes> + <include location="local" impldecl="in declaration">datablocks/rating.h</include> +</includes> +<variables> + <variable access="public">RatingList::iterator rating_it;</variable> +</variables> +<layoutdefaults spacing="6" margin="6"/> +<layoutfunctions spacing="KDialog::spacingHint" margin="KDialog::marginHint"/> +<includes> + <include location="global" impldecl="in implementation">kpushbutton.h</include> + <include location="global" impldecl="in implementation">tdelistview.h</include> +</includes> +</UI> diff --git a/src/widgets/ratingwidget.cpp b/src/widgets/ratingwidget.cpp new file mode 100644 index 0000000..10f35af --- /dev/null +++ b/src/widgets/ratingwidget.cpp @@ -0,0 +1,164 @@ +/*************************************************************************** + copyright : (C) 2003-2005 by Robby Stephenson + email : [email protected] + ***************************************************************************/ + +/*************************************************************************** + * * + * This program is free software; you can redistribute it and/or modify * + * it under the terms of version 2 of the GNU General Public License as * + * published by the Free Software Foundation; * + * * + ***************************************************************************/ + +#include "ratingwidget.h" + +#include <tdeglobal.h> // needed for KMAX +#include <kiconloader.h> +#include <kdebug.h> + +#include <tqintdict.h> +#include <tqlayout.h> + +namespace { + static const int RATING_WIDGET_MAX_STAR_SIZE = 24; +} + +const TQPixmap& RatingWidget::pixmap(const TQString& value_) { + static TQIntDict<TQPixmap> pixmaps; + if(pixmaps.isEmpty()) { + pixmaps.insert(-1, new TQPixmap()); + } + bool ok; + int n = value_.toInt(&ok); + if(!ok || n < 1 || n > 10) { + return *pixmaps[-1]; + } + if(pixmaps[n]) { + return *pixmaps[n]; + } + + TQString picName = TQString::fromLatin1("stars%1").arg(n); + TQPixmap* pix = new TQPixmap(UserIcon(picName)); + pixmaps.insert(n, pix); + return *pix; +} + +RatingWidget::RatingWidget(int stars, TQWidget* parent_, const char* name_/*=0*/) + : TQHBox(parent_, name_), m_currIndex(-1), m_min(0), m_max(stars*2) { + m_pixOn = UserIcon(TQString::fromLatin1("star_on")); + m_pixOff = UserIcon(TQString::fromLatin1("star_off")); + m_pixHalf = UserIcon(TQString::fromLatin1("star_half")); + setSpacing(0); + + // find maximum width and height + int w = KMAX(RATING_WIDGET_MAX_STAR_SIZE, KMAX(m_pixOn.width(), m_pixOff.width())); + int h = KMAX(RATING_WIDGET_MAX_STAR_SIZE, KMAX(m_pixOn.height(), m_pixOff.height())); + for(int i = 0; i < stars; ++i) { + TQLabel* l = new TQLabel(this); + l->setFixedSize(w, h); + m_widgets.append(l); + } + init(); + + TQBoxLayout* l = dynamic_cast<TQBoxLayout*>(layout()); + if(l) { + l->addStretch(1); + } +} + +void RatingWidget::init() { + m_total = KMIN(m_max/2, static_cast<int>(m_widgets.count())); + uint i = 0; + for( ; static_cast<int>(i) < m_total; ++i) { + m_widgets.at(i)->setPixmap(m_pixOff); + } + for( ; i < m_widgets.count(); ++i) { + m_widgets.at(i)->setPixmap(TQPixmap()); + } + update(); +} + +void RatingWidget::update() { + int i = 0; + for( ; i <= (m_currIndex-1)/2; ++i) { + m_widgets.at(i)->setPixmap(m_pixOn); + } + for( ; i < m_total; ++i) { + m_widgets.at(i)->setPixmap(m_pixOff); + } + + if ( m_currIndex % 2 == 0 ) { + m_widgets.at(m_currIndex/2)->setPixmap(m_pixHalf); + } + + TQHBox::update(); +} + +void RatingWidget::mousePressEvent(TQMouseEvent* event_) { + // only react to left button + if(event_->button() != TQt::LeftButton) { + return; + } + + int idx; + TQWidget* child = childAt(event_->pos()); + bool left = false; + if(child) { + TQRect child_geom_left_half = child->geometry(); + child_geom_left_half.setWidth(child_geom_left_half.width()/2); + if ( child_geom_left_half.contains(event_->pos()) ) + left = true; + + idx = m_widgets.findRef(static_cast<TQLabel*>(child)); + // if the widget is clicked beyond the maximum value, clear it + // remember total and min are values, but index is zero-based! + if(idx > m_total-1) { + idx = -1; + } else if(idx < m_min-1) { + idx = m_min-1; // limit to minimum, remember index is zero-based + } + } else { + idx = -1; + } + + int oldCurrent = m_currIndex; + + m_currIndex = idx*2+1; + + if ( left ) + m_currIndex--; + + if ( oldCurrent != m_currIndex ) { + update(); + emit modified(); + } +} + +void RatingWidget::clear() { + m_currIndex = -1; + update(); +} + +TQString RatingWidget::text() const { + // index is index of the list, which is zero-based. Add 1! + return m_currIndex == -1 ? TQString::null : TQString::number(double(m_currIndex+1)/2); +} + +void RatingWidget::setText(const TQString& text_) { + bool ok; + // text is value, subtract one to get index + m_currIndex =text_.toInt(&ok)-1; + if(ok) { + if(m_currIndex > m_total-1) { + m_currIndex = -1; + } else if(m_currIndex < m_min-1) { + m_currIndex = m_min-1; // limit to minimum, remember index is zero-based + } + } else { + m_currIndex = -1; + } + update(); +} + +#include "ratingwidget.moc" diff --git a/src/widgets/ratingwidget.h b/src/widgets/ratingwidget.h new file mode 100644 index 0000000..a603def --- /dev/null +++ b/src/widgets/ratingwidget.h @@ -0,0 +1,63 @@ +/*************************************************************************** + copyright : (C) 2003-2005 by Robby Stephenson + email : [email protected] + ***************************************************************************/ + +/*************************************************************************** + * * + * This program is free software; you can redistribute it and/or modify * + * it under the terms of version 2 of the GNU General Public License as * + * published by the Free Software Foundation; * + * * + ***************************************************************************/ + +#ifndef RATINGWIDGET_H +#define RATINGWIDGET_H + +#include <tqhbox.h> +#include <tqptrlist.h> +#include <tqlabel.h> +#include <tqpixmap.h> +#include <tqstringlist.h> + +/** + * @author Robby Stephenson + */ +class RatingWidget : public TQHBox { +TQ_OBJECT + +typedef TQPtrList<TQLabel> LabelList; + +public: + RatingWidget(int stars, TQWidget* parent, const char* name = 0); + + void clear(); + TQString text() const; + void setText(const TQString& text); + + static const TQPixmap& pixmap(const TQString& value); + +public slots: + void update(); + +signals: + void modified(); + +protected: + virtual void mousePressEvent(TQMouseEvent* e); + +private: + void init(); + + LabelList m_widgets; + + int m_currIndex; + int m_total; + int m_min; + int m_max; + + TQPixmap m_pixOn; + TQPixmap m_pixOff; + TQPixmap m_pixHalf; +}; +#endif diff --git a/src/widgets/recipelistview.cpp b/src/widgets/recipelistview.cpp new file mode 100644 index 0000000..536d06c --- /dev/null +++ b/src/widgets/recipelistview.cpp @@ -0,0 +1,447 @@ +/*************************************************************************** +* Copyright (C) 2004 by * +* Jason Kivlighn ([email protected]) * +* Unai Garro ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "recipelistview.h" + +#include <tqintdict.h> +#include <tqdatastream.h> +#include <tqtooltip.h> + +#include <tdeapplication.h> +#include <kdebug.h> +#include <tdeconfig.h> +#include <tdeglobal.h> +#include <tdelocale.h> +#include <kiconloader.h> +#include <kprogress.h> + +#include "backends/recipedb.h" + +class UncategorizedItem : public TQListViewItem +{ +public: + UncategorizedItem( TQListView *lv ) : TQListViewItem( lv, i18n("Uncategorized") ){} + int rtti() const { return 1006; } +}; + +RecipeItemDrag::RecipeItemDrag( RecipeListItem *recipeItem, TQWidget *dragSource, const char *name ) + : TQStoredDrag( RECIPEITEMMIMETYPE, dragSource, name ) +{ + if ( recipeItem ) { + TQByteArray data; + TQDataStream out( data, IO_WriteOnly ); + out << recipeItem->recipeID(); + out << recipeItem->title(); + setEncodedData( data ); + } +} + +bool RecipeItemDrag::canDecode( TQMimeSource* e ) +{ + return e->provides( RECIPEITEMMIMETYPE ); +} + +bool RecipeItemDrag::decode( const TQMimeSource* e, RecipeListItem& item ) +{ + if ( !e ) + return false; + + TQByteArray data = e->encodedData( RECIPEITEMMIMETYPE ); + if ( data.isEmpty() ) + return false; + + TQString title; + int recipeID; + TQDataStream in( data, IO_ReadOnly ); + in >> recipeID; + in >> title; + + item.setTitle( title ); + item.setRecipeID( recipeID ); + + return true; +} + +class RecipeListToolTip : public TQToolTip +{ +public: + RecipeListToolTip( RecipeListView *view ) : TQToolTip(view->viewport()), m_view(view) + {} + + void maybeTip( const TQPoint &point ) + { + TQListViewItem *item = m_view->itemAt( point ); + if ( item ) { + TQString text = m_view->tooltip(item,0); + if ( !text.isEmpty() ) + tip( m_view->itemRect( item ), text ); + } + } + +private: + RecipeListView *m_view; + +}; + + +RecipeListView::RecipeListView( TQWidget *parent, RecipeDB *db ) : StdCategoryListView( parent, db ), + flat_list( false ), + m_uncat_item(0), + m_progress_dlg(0) +{ + setColumnText( 0, i18n( "Recipe" ) ); + + TDEConfig *config = TDEGlobal::config(); config->setGroup( "Performance" ); + curr_limit = config->readNumEntry("CategoryLimit",-1); + + TDEIconLoader il; + setPixmap( il.loadIcon( "categories", TDEIcon::NoGroup, 16 ) ); + + setSelectionMode( TQListView::Extended ); + + (void)new RecipeListToolTip(this); +} + +void RecipeListView::init() +{ + connect( database, TQ_SIGNAL( recipeCreated( const Element &, const ElementList & ) ), TQ_SLOT( createRecipe( const Element &, const ElementList & ) ) ); + connect( database, TQ_SIGNAL( recipeRemoved( int ) ), TQ_SLOT( removeRecipe( int ) ) ); + connect( database, TQ_SIGNAL( recipeRemoved( int, int ) ), TQ_SLOT( removeRecipe( int, int ) ) ); + connect( database, TQ_SIGNAL( recipeModified( const Element &, const ElementList & ) ), TQ_SLOT( modifyRecipe( const Element &, const ElementList & ) ) ); + + StdCategoryListView::init(); +} + +TQDragObject *RecipeListView::dragObject() +{ + RecipeListItem * item = dynamic_cast<RecipeListItem*>( currentItem() ); + if ( item != 0 ) { + RecipeItemDrag * obj = new RecipeItemDrag( item, this, "Recipe drag item" ); + /*const TQPixmap *pm = item->pixmap(0); + if( pm ) + obj->setPixmap( *pm );*/ + return obj; + } + return 0; +} + +bool RecipeListView::acceptDrag( TQDropEvent *event ) const +{ + return RecipeItemDrag::canDecode( event ); +} + +TQString RecipeListView::tooltip(TQListViewItem *item, int /*column*/) const +{ + if ( item->rtti() == RECIPELISTITEM_RTTI ) { + RecipeListItem *recipe_it = (RecipeListItem*)item; + + Recipe r; + database->loadRecipe(&r,RecipeDB::Meta|RecipeDB::Noatime,recipe_it->recipeID() ); + + TDELocale *locale = TDEGlobal::locale(); + + return TQString("<center><b>%7</b></center><center>__________</center>%1 %2<br />%3 %4<br />%5 %6") + .arg(i18n("Created:")).arg(locale->formatDateTime(r.ctime)) + .arg(i18n("Modified:")).arg(locale->formatDateTime(r.mtime)) + .arg(i18n("Last Accessed:")).arg(locale->formatDateTime(r.atime)) + .arg(recipe_it->title()); + }/* Maybe this would be handy + else if ( item->rtti() == CATEGORYLISTITEM_RTTI ) { + CategoryListItem *cat_it = (CategoryListItem*)item; + + return TQString("<b>%1</b><hr />%2: %3") + .arg(cat_it->categoryName()) + .arg(i18n("Recipes")) + .arg(TQString::number(WHATEVER THE CHILD COUNT IS)); + }*/ + + return TQString::null; +} + +void RecipeListView::load(int limit, int offset) +{ + m_uncat_item = 0; + + if ( flat_list ) { + ElementList recipeList; + database->loadRecipeList( &recipeList ); + + ElementList::const_iterator recipe_it; + for ( recipe_it = recipeList.begin();recipe_it != recipeList.end();++recipe_it ) { + Recipe recipe; + recipe.recipeID = ( *recipe_it ).id; + recipe.title = ( *recipe_it ).name; + createRecipe( recipe, -1 ); + } + } + else { + StdCategoryListView::load(limit,offset); + + if ( offset == 0 ) { + ElementList recipeList; + database->loadUncategorizedRecipes( &recipeList ); + + ElementList::const_iterator recipe_it; + for ( recipe_it = recipeList.begin();recipe_it != recipeList.end();++recipe_it ) { + Recipe recipe; + recipe.recipeID = ( *recipe_it ).id; + recipe.title = ( *recipe_it ).name; + createRecipe( recipe, -1 ); + } + } + } +} + +void RecipeListView::populate( TQListViewItem *item ) +{ + CategoryItemInfo *cat_item = dynamic_cast<CategoryItemInfo*>(item); + if ( !cat_item || cat_item->isPopulated() ) return; + + delete item->firstChild(); //delete the "pseudo item" + + if ( m_progress_dlg ){ + m_progress_dlg->progressBar()->advance(1); + kapp->processEvents(); + } + + StdCategoryListView::populate(item); + + if ( !flat_list ) { + int id = cat_item->categoryId(); + + // Now show the recipes + ElementList recipeList; + database->loadRecipeList( &recipeList, id ); + + ElementList::const_iterator recipe_it; + for ( recipe_it = recipeList.begin(); recipe_it != recipeList.end(); ++recipe_it ) { + Recipe recipe; + recipe.recipeID = ( *recipe_it ).id; + recipe.title = ( *recipe_it ).name; + createRecipe( recipe, id ); + } + } +} + +void RecipeListView::populateAll( TQListViewItem *parent ) +{ + bool first = false; + if ( !parent ) { + first = true; + m_progress_dlg = new KProgressDialog(this,"populate_all_prog_dlg",TQString::null,i18n("Loading recipes"),true); + m_progress_dlg->setAllowCancel(false); + m_progress_dlg->progressBar()->setTotalSteps(0); + m_progress_dlg->progressBar()->setPercentageVisible(false); + + m_progress_dlg->grabKeyboard(); //don't let the user keep hitting keys + + parent = firstChild(); + } + else { + populate( parent ); + parent = parent->firstChild(); + } + + for ( TQListViewItem *item = parent; item; item = item->nextSibling() ) { + if ( m_progress_dlg && m_progress_dlg->wasCancelled() ) + break; + + populateAll( item ); + } + + if ( first ) { + delete m_progress_dlg; + m_progress_dlg = 0; + } +} + +void RecipeListView::createRecipe( const Recipe &recipe, int parent_id ) +{ + if ( parent_id == -1 ) { + if ( !m_uncat_item && curr_offset == 0 ) { + m_uncat_item = new UncategorizedItem(this); + if ( childCount() == 1 ) //only call createElement if this is the only item in the list + createElement(m_uncat_item); //otherwise, this item won't stay at the top + } + + if ( m_uncat_item ) + new RecipeListItem( m_uncat_item, recipe ); + } + else { + CategoryListItem *parent = (CategoryListItem*)items_map[ parent_id ]; + if ( parent && parent->isPopulated() ) + createElement(new RecipeListItem( parent, recipe )); + } +} + +void RecipeListView::createRecipe( const Element &recipe_el, const ElementList &categories ) +{ + Recipe recipe; + recipe.recipeID = recipe_el.id; + recipe.title = recipe_el.name; + + if ( categories.count() == 0 ) { + createRecipe( recipe, -1 ); + } + else { + for ( ElementList::const_iterator cat_it = categories.begin(); cat_it != categories.end(); ++cat_it ) { + int cur_cat_id = ( *cat_it ).id; + + TQListViewItemIterator iterator( this ); + while ( iterator.current() ) { + if ( iterator.current() ->rtti() == 1001 ) { + CategoryListItem * cat_item = ( CategoryListItem* ) iterator.current(); + if ( cat_item->categoryId() == cur_cat_id ) { + createRecipe( recipe, cur_cat_id ); + } + } + ++iterator; + } + } + } +} + +void RecipeListView::createElement( TQListViewItem *item ) +{ + CategoryItemInfo *cat_item = dynamic_cast<CategoryItemInfo*>(item); + if ( cat_item && !cat_item->isPopulated() ) { + new PseudoListItem( item ); + } + + //if ( cat_item && !cat_item->isPopulated() && item->rtti() == RECIPELISTITEM_RTTI ) + // return; + + #if 0 + ElementList list; + database->loadRecipeList( &list, cat_item->categoryId() ); + if ( list.count() > 0 ) + #endif + + CategoryListView::createElement(item); +} + +void RecipeListView::modifyRecipe( const Element &recipe, const ElementList &categories ) +{ + removeRecipe( recipe.id ); + createRecipe( recipe, categories ); +} + +void RecipeListView::removeRecipe( int id ) +{ + TQListViewItemIterator iterator( this ); + while ( iterator.current() ) { + if ( iterator.current() ->rtti() == 1000 ) { + RecipeListItem * recipe_it = ( RecipeListItem* ) iterator.current(); + if ( recipe_it->recipeID() == id ) { + removeElement(recipe_it); + + //delete the "Uncategorized" item if we removed the last recipe that was under it + if ( m_uncat_item && m_uncat_item->childCount() == 0 ) { + delete m_uncat_item; + m_uncat_item = 0; + } + } + } + ++iterator; + } +} + +void RecipeListView::removeRecipe( int recipe_id, int cat_id ) +{ + TQListViewItem * item = items_map[ cat_id ]; + + //find out if this is the only category the recipe belongs to + int finds = 0; + TQListViewItemIterator iterator( this ); + while ( iterator.current() ) { + if ( iterator.current() ->rtti() == 1000 ) { + RecipeListItem * recipe_it = ( RecipeListItem* ) iterator.current(); + + if ( recipe_it->recipeID() == recipe_id ) { + if ( finds > 1 ) + break; + finds++; + } + } + ++iterator; + } + + //do this to only iterate over children of 'item' + TQListViewItem *pEndItem = NULL; + TQListViewItem *pStartItem = item; + do { + if ( pStartItem->nextSibling() ) + pEndItem = pStartItem->nextSibling(); + else + pStartItem = pStartItem->parent(); + } + while ( pStartItem && !pEndItem ); + + iterator = TQListViewItemIterator( item ); + while ( iterator.current() != pEndItem ) { + if ( iterator.current() ->rtti() == 1000 ) { + RecipeListItem * recipe_it = ( RecipeListItem* ) iterator.current(); + + if ( recipe_it->recipeID() == recipe_id ) { + + if ( finds == 1 ) { + //the item is now uncategorized + if ( !m_uncat_item && curr_offset == 0 ) + m_uncat_item = new UncategorizedItem(this); + if ( m_uncat_item ) { + Recipe r; + r.title = recipe_it->title(); r.recipeID = recipe_id; + new RecipeListItem(m_uncat_item,r); + } + } + removeElement(recipe_it); + break; + } + } + ++iterator; + } +} + +void RecipeListView::removeCategory( int id ) +{ + TQListViewItem * item = items_map[ id ]; + if ( !item ) + return ; //this may have been deleted already by its parent being deleted + + moveChildrenToRoot( item ); + + StdCategoryListView::removeCategory( id ); +} + +void RecipeListView::moveChildrenToRoot( TQListViewItem *item ) +{ + TQListViewItem * next_sibling; + for ( TQListViewItem * it = item->firstChild(); it; it = next_sibling ) { + next_sibling = it->nextSibling(); + if ( it->rtti() == 1000 ) { + RecipeListItem *recipe_it = (RecipeListItem*) it; + Recipe r; + r.title = recipe_it->title(); r.recipeID = recipe_it->recipeID(); + + //the item is now uncategorized + removeElement(it,false); + it->parent() ->takeItem( it ); + if ( !m_uncat_item && curr_offset == 0 ) + m_uncat_item = new UncategorizedItem(this); + if ( m_uncat_item ) + new RecipeListItem(m_uncat_item,r); + } + moveChildrenToRoot( it ); + delete it; + } +} + +#include "recipelistview.moc" diff --git a/src/widgets/recipelistview.h b/src/widgets/recipelistview.h new file mode 100644 index 0000000..642f768 --- /dev/null +++ b/src/widgets/recipelistview.h @@ -0,0 +1,166 @@ +/*************************************************************************** +* Copyright (C) 2004 by * +* Jason Kivlighn ([email protected]) * +* Unai Garro ([email protected]) * +* * +* Copyright (C) 2006 Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef RECIPELISTVIEW_H +#define RECIPELISTVIEW_H + +#include <tqdragobject.h> + +#include "categorylistview.h" +#include "datablocks/recipe.h" + +class TQDragObject; +class TQDropEvent; + +class KProgressDialog; + +class RecipeDB; + +#define RECIPELISTITEM_RTTI 1000 + +#define RECIPEITEMMIMETYPE "data/x-kde.recipe.item" + +class RecipeListItem : public TQListViewItem +{ +public: + RecipeListItem( TQListView* qlv, const Recipe &r ) : TQListViewItem( qlv ) + { + init( r ); + } + + RecipeListItem( TQListView* qlv, TQListViewItem *after, const Recipe &r ) : TQListViewItem( qlv, after ) + { + init( r ); + } + + RecipeListItem( CategoryListItem* it, const Recipe &r ) : TQListViewItem( it ) + { + init( r ); + } + + RecipeListItem( CategoryListItem* it, TQListViewItem *after, const Recipe &r ) : TQListViewItem( it, after ) + { + init( r ); + } + + RecipeListItem( TQListViewItem* it, const Recipe &r ) : TQListViewItem( it ) + { + init( r ); + } + + int rtti() const + { + return RECIPELISTITEM_RTTI; + } + + ~RecipeListItem( void ) + { + delete recipeStored; + } + + int recipeID() const + { + return recipeStored->recipeID; + } + TQString title() const + { + return recipeStored->title; + } + + void setRecipeID( int id ) + { + recipeStored->recipeID = id; + } + void setTitle( const TQString &title ) + { + recipeStored->title = title; + } + +protected: + Recipe *recipeStored; + +public: + virtual TQString text( int column ) const + { + switch ( column ) { + case 0: + return ( recipeStored->title ); + break; + case 1: + return ( TQString::number( recipeStored->recipeID ) ); + break; + default: + return ( TQString::null ); + } + } + +private: + void init( const Recipe &r ) + { + recipeStored = new Recipe(); + + //note: we only store the title and id + recipeStored->recipeID = r.recipeID; + recipeStored->title = r.title; + } +}; + +class RecipeItemDrag : public TQStoredDrag +{ +public: + RecipeItemDrag( RecipeListItem *recipeItem, TQWidget *dragSource = 0, const char *name = 0 ); + static bool canDecode( TQMimeSource* e ); + static bool decode( const TQMimeSource* e, RecipeListItem& item ); +}; + +class RecipeListView : public StdCategoryListView +{ + TQ_OBJECT + +public: + RecipeListView( TQWidget *parent, RecipeDB *db ); + +public slots: + void populateAll( TQListViewItem *parent = 0 ); + +protected slots: + virtual void createRecipe( const Recipe &, int parent_id ); + virtual void createRecipe( const Element &recipe, const ElementList &categories ); + virtual void modifyRecipe( const Element &recipe, const ElementList &categories ); + virtual void removeRecipe( int ); + virtual void removeRecipe( int, int ); + +protected: + virtual void init(); + virtual void createElement( TQListViewItem * ); + virtual void removeCategory( int id ); + virtual TQDragObject *dragObject(); + virtual bool acceptDrag( TQDropEvent *event ) const; + virtual void populate( TQListViewItem *item ); + virtual TQString tooltip(TQListViewItem *item, int column) const; + + friend class RecipeListToolTip; + + void load(int limit, int offset); + +private: + void moveChildrenToRoot( TQListViewItem * ); + + bool flat_list; + TQListViewItem *m_uncat_item; + TQListViewItem *lastElementCurrLevel; + + KProgressDialog *m_progress_dlg; +}; + +#endif //RECIPELISTVIEW_H diff --git a/src/widgets/unitcombobox.cpp b/src/widgets/unitcombobox.cpp new file mode 100644 index 0000000..526f61a --- /dev/null +++ b/src/widgets/unitcombobox.cpp @@ -0,0 +1,145 @@ +/*************************************************************************** +* Copyright (C) 2006 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "unitcombobox.h" + +#include <tqlistbox.h> + +#include <tdelocale.h> +#include <tdeconfig.h> +#include <tdeglobal.h> + +#include "backends/recipedb.h" +#include "datablocks/elementlist.h" + +UnitComboBox::UnitComboBox( TQWidget *parent, RecipeDB *db, Unit::Type type ) : KComboBox( parent ), + database( db ), m_type(type) +{ + connect( database, TQ_SIGNAL( unitCreated( const Unit & ) ), TQ_SLOT( createUnit( const Unit & ) ) ); + connect( database, TQ_SIGNAL( unitRemoved( int ) ), TQ_SLOT( removeUnit( int ) ) ); +} + +void UnitComboBox::popup() +{ + if ( count() == 1 ) + reload(); + KComboBox::popup(); +} + +Unit UnitComboBox::unit() const +{ + Unit u; + u.name = currentText(); + u.id = id(currentItem()); + return u; +} + +void UnitComboBox::reload() +{ + TQString remember_filter = currentText(); + + UnitList unitList; + database->loadUnits( &unitList, m_type ); + + clear(); + unitComboRows.clear(); + + //Now load the categories + loadUnits(unitList); + + if ( listBox()->findItem( remember_filter, TQt::ExactMatch ) ) { + setCurrentText( remember_filter ); + } +} + +void UnitComboBox::loadUnits( const UnitList &unitList ) +{ + int row = 0; + for ( UnitList::const_iterator it = unitList.begin(); it != unitList.end(); ++it ) { + insertItem( (*it).name ); + unitComboRows.insert( row, (*it).id ); // store unit id's in the combobox position to obtain the unit id later + row++; + } +} + +void UnitComboBox::setSelected( int unitID ) +{ + //do a reverse lookup on the row->id map + TQMap<int, int>::const_iterator it; + for ( it = unitComboRows.begin(); it != unitComboRows.end(); ++it ) { + if ( it.data() == unitID ) { + setCurrentItem(it.key()); + break; + } + } +} + +int UnitComboBox::id( int row ) const +{ + return unitComboRows[ row ]; +} + +void UnitComboBox::createUnit( const Unit &element ) +{ + int row = findInsertionPoint( element.name ); + + insertItem( element.name, row ); + + //now update the map by pushing everything after this item down + TQMap<int, int> new_map; + for ( TQMap<int, int>::iterator it = unitComboRows.begin(); it != unitComboRows.end(); ++it ) { + if ( it.key() >= row ) { + new_map.insert( it.key() + 1, it.data() ); + } + else + new_map.insert( it.key(), it.data() ); + } + unitComboRows = new_map; + unitComboRows.insert( row, element.id ); +} + +void UnitComboBox::removeUnit( int id ) +{ + int row = -1; + for ( TQMap<int, int>::iterator it = unitComboRows.begin(); it != unitComboRows.end(); ++it ) { + if ( it.data() == id ) { + row = it.key(); + removeItem( row ); + unitComboRows.remove( it ); + break; + } + } + + if ( row == -1 ) + return ; + + //now update the map by pushing everything after this item up + TQMap<int, int> new_map; + for ( TQMap<int, int>::iterator it = unitComboRows.begin(); it != unitComboRows.end(); ++it ) { + if ( it.key() > row ) { + new_map.insert( it.key() - 1, it.data() ); + } + else + new_map.insert( it.key(), it.data() ); + } + unitComboRows = new_map; +} + +int UnitComboBox::findInsertionPoint( const TQString &name ) +{ + for ( int i = 1; i < count(); i++ ) { + if ( TQString::localeAwareCompare( name, text( i ) ) < 0 ) + return i; + } + + return count(); +} + +#include "unitcombobox.moc" diff --git a/src/widgets/unitcombobox.h b/src/widgets/unitcombobox.h new file mode 100644 index 0000000..dd75107 --- /dev/null +++ b/src/widgets/unitcombobox.h @@ -0,0 +1,52 @@ +/*************************************************************************** +* Copyright (C) 2006 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef UNITCOMBOBOX_H +#define UNITCOMBOBOX_H + +#include <kcombobox.h> + +#include <tqmap.h> + +#include "datablocks/unit.h" + +class RecipeDB; + +class UnitComboBox : public KComboBox +{ + TQ_OBJECT + +public: + UnitComboBox( TQWidget *parent, RecipeDB *db, Unit::Type type = Unit::All ); + + void reload(); + int id( int row ) const; + void setSelected( int unitID ); + Unit unit() const; + +protected: + virtual void popup(); + +private slots: + void createUnit( const Unit & ); + void removeUnit( int id ); + + int findInsertionPoint( const TQString &name ); + +private: + void loadUnits( const UnitList &unitList ); + + RecipeDB *database; + TQMap<int, int> unitComboRows; // Contains the unit id for every given row in the unit combobox + Unit::Type m_type; +}; + +#endif //UNITCOMBOBOX_H + diff --git a/src/widgets/unitlistview.cpp b/src/widgets/unitlistview.cpp new file mode 100644 index 0000000..9c70b1b --- /dev/null +++ b/src/widgets/unitlistview.cpp @@ -0,0 +1,369 @@ +/*************************************************************************** +* Copyright (C) 2004 by * +* Jason Kivlighn ([email protected]) * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* * +* Copyright (C) 2006 Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "unitlistview.h" + +#include <tqcombobox.h> +#include <tqheader.h> + +#include <tdemessagebox.h> +#include <tdeconfig.h> +#include <tdelocale.h> +#include <tdeglobal.h> +#include <kiconloader.h> +#include <tdepopupmenu.h> +#include <kdebug.h> + +#include "backends/recipedb.h" +#include "dialogs/createunitdialog.h" +#include "dialogs/dependanciesdialog.h" +#include "datablocks/unit.h" + +class UnitListViewItem : public TQListViewItem +{ +public: + UnitListViewItem( TQListView* qlv, const Unit &u ) : TQListViewItem( qlv ), m_unit(u) + { + updateType(m_unit.type); + } + + virtual TQString text( int column ) const + { + switch ( column ) { + case 0: return m_unit.name; + case 1: return m_unit.name_abbrev; + case 2: return m_unit.plural; + case 3: return m_unit.plural_abbrev; + case 4: return m_type; + case 5: return TQString::number(m_unit.id); + default: return TQString::null; + } + } + + void setType( Unit::Type type ){ m_unit.type = type; updateType(type); } + + Unit unit() const { return m_unit; }; + void setUnit( const Unit &u ) { m_unit = u; } + +protected: + virtual void setText( int column, const TQString &text ) { + switch ( column ) { + case 0: m_unit.name = text; break; + case 1: m_unit.name_abbrev = text; break; + case 2: m_unit.plural = text; break; + case 3: m_unit.plural_abbrev = text; break; + } + } + +private: + void updateType( Unit::Type t ) { + switch ( t ) { + case Unit::Other: m_type = i18n("Other"); break; + case Unit::Mass: m_type = i18n("Mass"); break; + case Unit::Volume: m_type = i18n("Volume"); break; + default: break; + } + } + + Unit m_unit; + TQString m_type; +}; + +UnitListView::UnitListView( TQWidget *parent, RecipeDB *db ) : DBListViewBase( parent,db,db->unitCount() ) +{ + setAllColumnsShowFocus( true ); + setDefaultRenameAction( TQListView::Reject ); +} + +void UnitListView::init() +{ + connect( database, TQ_SIGNAL( unitCreated( const Unit & ) ), TQ_SLOT( checkCreateUnit( const Unit & ) ) ); + connect( database, TQ_SIGNAL( unitRemoved( int ) ), TQ_SLOT( removeUnit( int ) ) ); +} + +void UnitListView::load( int limit, int offset ) +{ + UnitList unitList; + database->loadUnits( &unitList, Unit::All, limit, offset ); + + for ( UnitList::const_iterator it = unitList.begin(); it != unitList.end(); ++it ) { + if ( !( *it ).name.isEmpty() || !( *it ).plural.isEmpty() ) + createUnit( *it ); + } +} + +void UnitListView::checkCreateUnit( const Unit &el ) +{ + if ( handleElement(el.name) ) { //only create this unit if the base class okays it + createUnit(el); + } +} + + +StdUnitListView::StdUnitListView( TQWidget *parent, RecipeDB *db, bool editable ) : UnitListView( parent, db ) +{ + addColumn( i18n( "Unit" ) ); + addColumn( i18n( "Abbreviation" ) ); + addColumn( i18n( "Plural" ) ); + addColumn( i18n( "Abbreviation" ) ); + addColumn( i18n( "Type" ) ); + + TDEConfig * config = TDEGlobal::config(); + config->setGroup( "Advanced" ); + bool show_id = config->readBoolEntry( "ShowID", false ); + addColumn( i18n( "Id" ), show_id ? -1 : 0 ); + + if ( editable ) { + setRenameable( 0, true ); + setRenameable( 1, true ); + setRenameable( 2, true ); + setRenameable( 3, true ); + setRenameable( 4, true ); + + TDEIconLoader il; + + kpop = new TDEPopupMenu( this ); + kpop->insertItem( il.loadIcon( "document-new", TDEIcon::NoGroup, 16 ), i18n( "&Create" ), this, TQ_SLOT( createNew() ), CTRL + Key_C ); + kpop->insertItem( il.loadIcon( "edit-delete", TDEIcon::NoGroup, 16 ), i18n( "&Delete" ), this, TQ_SLOT( remove + () ), Key_Delete ); + kpop->insertItem( il.loadIcon( "edit", TDEIcon::NoGroup, 16 ), i18n( "&Rename" ), this, TQ_SLOT( rename() ), CTRL + Key_R ); + kpop->polish(); + + typeComboBox = new TQComboBox( false, viewport() ); + typeComboBox->insertItem(i18n("Other")); + typeComboBox->insertItem(i18n("Mass")); + typeComboBox->insertItem(i18n("Volume")); + addChild( typeComboBox ); + typeComboBox->hide(); + + connect( typeComboBox, TQ_SIGNAL( activated(int) ), TQ_SLOT( updateType(int) ) ); + connect( this, TQ_SIGNAL( selectionChanged() ), TQ_SLOT( hideTypeCombo() ) ); + + connect( this, TQ_SIGNAL( contextMenu( TDEListView *, TQListViewItem *, const TQPoint & ) ), TQ_SLOT( showPopup( TDEListView *, TQListViewItem *, const TQPoint & ) ) ); + connect( this, TQ_SIGNAL( doubleClicked( TQListViewItem*, const TQPoint &, int ) ), this, TQ_SLOT( modUnit( TQListViewItem*, const TQPoint &, int ) ) ); + connect( this, TQ_SIGNAL( itemRenamed( TQListViewItem*, const TQString &, int ) ), this, TQ_SLOT( saveUnit( TQListViewItem*, const TQString &, int ) ) ); + } +} + +void StdUnitListView::insertTypeComboBox( TQListViewItem* it ) +{ + TQRect r; + + // Constraints Box1 + r = header() ->sectionRect( 4 ); //start at the section 2 header + r.moveBy( 0, itemRect( it ).y() ); //Move down to the item, note that its height is same as header's right now. + + r.setHeight( it->height() ); // Set the item's height + r.setWidth( header() ->sectionRect( 4 ).width() ); // and width + typeComboBox->setGeometry( r ); + + UnitListViewItem *unit_it = (UnitListViewItem*)it; + typeComboBox->setCurrentItem( unit_it->unit().type ); + + typeComboBox->show(); +} + +void StdUnitListView::updateType( int type ) +{ + UnitListViewItem *unit_it = (UnitListViewItem*)currentItem(); + unit_it->setType((Unit::Type)type); + + database->modUnit( unit_it->unit() ); +} + +void StdUnitListView::hideTypeCombo() +{ + typeComboBox->hide(); +} + +void StdUnitListView::showPopup( TDEListView * /*l*/, TQListViewItem *i, const TQPoint &p ) +{ + if ( i ) + kpop->exec( p ); +} + +void StdUnitListView::createNew() +{ + CreateUnitDialog * unitDialog = new CreateUnitDialog( this ); + + if ( unitDialog->exec() == TQDialog::Accepted ) { + Unit result = unitDialog->newUnit(); + + //check bounds first + if ( checkBounds( result ) + && database->findExistingUnitByName( result.name ) == -1 + && database->findExistingUnitByName( result.plural ) == -1 + ) { + database->createNewUnit( result ); + } + } + delete unitDialog; +} + +void StdUnitListView::remove() +{ + // Find selected unit item + UnitListViewItem* it = (UnitListViewItem*)currentItem(); + + if ( it ) { + int unitID = it->unit().id; + + ElementList recipeDependancies, propertyDependancies, weightDependancies; + database->findUnitDependancies( unitID, &propertyDependancies, &recipeDependancies, &weightDependancies ); + + TQValueList<ListInfo> lists; + if ( !recipeDependancies.isEmpty() ) { + ListInfo info; + info.list = recipeDependancies; + info.name = i18n("Recipes"); + lists << info; + } + if ( !propertyDependancies.isEmpty() ) { + ListInfo info; + info.list = propertyDependancies; + info.name = i18n("Properties"); + lists << info; + } + if ( !weightDependancies.isEmpty() ) { + ListInfo info; + info.list = weightDependancies; + info.name = i18n("Ingredient Weights"); + lists << info; + } + + if ( lists.isEmpty() ) + database->removeUnit( unitID ); + else { // need warning! + DependanciesDialog warnDialog( this, lists ); + if ( !recipeDependancies.isEmpty() ) + warnDialog.setCustomWarning( i18n("You are about to permanantly delete recipes from your database.") ); + if ( warnDialog.exec() == TQDialog::Accepted ) + database->removeUnit( unitID ); + } + } +} + +void StdUnitListView::rename() +{ + TQListViewItem * item = currentItem(); + + if ( item ) { + CreateUnitDialog unitDialog( this, item->text(0), item->text(2), item->text(1), item->text(3), false ); + + if ( unitDialog.exec() == TQDialog::Accepted ) { + UnitListViewItem *unit_item = (UnitListViewItem*)item; + Unit origUnit = unit_item->unit(); + Unit newUnit = unitDialog.newUnit(); + + //for each changed entry, save the change individually + + Unit unit = origUnit; + + if ( newUnit.name != origUnit.name ) { + unit.name = newUnit.name; + unit_item->setUnit( unit ); + saveUnit( unit_item, newUnit.name, 0 ); + + //saveUnit will call database->modUnit which deletes the list item we were using + unit_item = (UnitListViewItem*) findItem( TQString::number(unit.id), 5 ); + } + + if ( newUnit.plural != origUnit.plural ) { + unit.plural = newUnit.plural; + unit_item->setUnit( unit ); + saveUnit( unit_item, newUnit.plural, 2 ); + unit_item = (UnitListViewItem*) findItem( TQString::number(unit.id), 5 ); + } + + if ( !newUnit.name_abbrev.stripWhiteSpace().isEmpty() && newUnit.name_abbrev != origUnit.name_abbrev ) { + unit.name_abbrev = newUnit.name_abbrev; + unit_item->setUnit( unit ); + saveUnit( unit_item, newUnit.name_abbrev, 1 ); + unit_item = (UnitListViewItem*) findItem( TQString::number(unit.id), 5 ); + } + if ( !newUnit.plural_abbrev.stripWhiteSpace().isEmpty() && newUnit.plural_abbrev != origUnit.plural_abbrev ) { + unit.plural_abbrev = newUnit.plural_abbrev; + unit_item->setUnit( unit ); + saveUnit( unit_item, newUnit.plural_abbrev, 3 ); + } + } + } +} + +void StdUnitListView::createUnit( const Unit &unit ) +{ + createElement(new UnitListViewItem( this, unit )); +} + +void StdUnitListView::removeUnit( int id ) +{ + TQListViewItem * item = findItem( TQString::number( id ), 5 ); + removeElement(item); +} + +void StdUnitListView::modUnit( TQListViewItem* i, const TQPoint & /*p*/, int c ) +{ + if ( i ) { + if ( c != 4 ) + UnitListView::rename( i, c ); + else { + insertTypeComboBox(i); + } + } +} + +void StdUnitListView::saveUnit( TQListViewItem* i, const TQString &text, int c ) +{ + //skip abbreviations + if ( c == 0 || c == 2 ) { + if ( !checkBounds( Unit( text, text ) ) ) { + reload(ForceReload); //reset the changed text + return ; + } + } + + int existing_id = database->findExistingUnitByName( text ); + + UnitListViewItem *unit_it = (UnitListViewItem*)i; + int unit_id = unit_it->unit().id; + if ( existing_id != -1 && existing_id != unit_id && !text.stripWhiteSpace().isEmpty() ) { //unit already exists with this label... merge the two + switch ( KMessageBox::warningContinueCancel( this, i18n( "This unit already exists. Continuing will merge these two units into one. Are you sure?" ) ) ) { + case KMessageBox::Continue: { + database->modUnit( unit_it->unit() ); + database->mergeUnits( unit_id, existing_id ); + break; + } + default: + reload(ForceReload); + break; + } + } + else { + database->modUnit( unit_it->unit() ); + } +} + +bool StdUnitListView::checkBounds( const Unit &unit ) +{ + if ( unit.name.length() > uint(database->maxUnitNameLength()) || unit.plural.length() > uint(database->maxUnitNameLength()) ) { + KMessageBox::error( this, TQString( i18n( "Unit name cannot be longer than %1 characters." ) ).arg( database->maxUnitNameLength() ) ); + return false; + } + else if ( unit.name.stripWhiteSpace().isEmpty() || unit.plural.stripWhiteSpace().isEmpty() ) + return false; + + return true; +} + +#include "unitlistview.moc" diff --git a/src/widgets/unitlistview.h b/src/widgets/unitlistview.h new file mode 100644 index 0000000..b71b67f --- /dev/null +++ b/src/widgets/unitlistview.h @@ -0,0 +1,76 @@ +/*************************************************************************** +* Copyright (C) 2004 by * +* Jason Kivlighn ([email protected]) * +* Unai Garro ([email protected]) * +* Cyril Bosselut ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef UNITLISTVIEW_H +#define UNITLISTVIEW_H + +#include "dblistviewbase.h" + +#include "datablocks/unit.h" + +class TQComboBox; + +class RecipeDB; +class TDEPopupMenu; + +class UnitListView : public DBListViewBase +{ + TQ_OBJECT + +public: + UnitListView( TQWidget *parent, RecipeDB *db ); + +public slots: + virtual void load( int curr_limit, int curr_offset ); + +protected slots: + virtual void createUnit( const Unit & ) = 0; + virtual void removeUnit( int ) = 0; + + void checkCreateUnit( const Unit &el ); + +protected: + virtual void init(); +}; + +class StdUnitListView : public UnitListView +{ + TQ_OBJECT + +public: + StdUnitListView( TQWidget *parent, RecipeDB *db, bool editable = false ); + +protected: + virtual void createUnit( const Unit & ); + virtual void removeUnit( int ); + +private slots: + void showPopup( TDEListView *, TQListViewItem *, const TQPoint & ); + void hideTypeCombo(); + void updateType( int type ); + + void createNew(); + void remove(); + void rename(); + + void modUnit( TQListViewItem* i, const TQPoint &p, int c ); + void saveUnit( TQListViewItem* i, const TQString &text, int c ); + +private: + bool checkBounds( const Unit &unit ); + void insertTypeComboBox( TQListViewItem* ); + + TDEPopupMenu *kpop; + TQComboBox *typeComboBox; +}; + +#endif //UNITLISTVIEW_H diff --git a/src/widgets/weightinput.cpp b/src/widgets/weightinput.cpp new file mode 100644 index 0000000..17d3d0a --- /dev/null +++ b/src/widgets/weightinput.cpp @@ -0,0 +1,63 @@ +/*************************************************************************** +* Copyright (C) 2006 by * +* Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#include "weightinput.h" + +#include <tdelocale.h> + +#include "backends/recipedb.h" +#include "datablocks/element.h" +#include "datablocks/weight.h" +#include "prepmethodcombobox.h" + +WeightInput::WeightInput( TQWidget *parent, RecipeDB *database, Unit::Type type, MixedNumber::Format format ) : + AmountUnitInput(parent,database,type,format), + m_database(database) +{ + prepMethodBox = new PrepMethodComboBox(false,this,database,i18n("-No Preparation-")); + prepMethodBox->reload(); + + connect( prepMethodBox, TQ_SIGNAL(activated(int)), TQ_SLOT(emitValueChanged()) ); +} + +void WeightInput::emitValueChanged() +{ + Weight w; + w.perAmount = amount().toDouble(); + + Unit u = unit(); + w.perAmountUnitID = u.id; + w.perAmountUnit = (w.perAmount>1)?u.plural:u.name; + + Element prep; + w.prepMethod = prep.name; + w.prepMethodID = prep.id; + emit valueChanged( w ); +} + +void WeightInput::setPrepMethod( const Element &prep ) +{ + if ( prep.id == -1 ) + prepMethodBox->setCurrentItem(0); + else + prepMethodBox->setSelected( prep.id ); + +} + +Element WeightInput::prepMethod() const +{ + Element prep; + prep.id = prepMethodBox->id( prepMethodBox->currentItem() ); + if ( prep.id != -1 ) + prep.name = prepMethodBox->currentText(); + return prep; +} + +#include "weightinput.moc" diff --git a/src/widgets/weightinput.h b/src/widgets/weightinput.h new file mode 100644 index 0000000..83ff6b3 --- /dev/null +++ b/src/widgets/weightinput.h @@ -0,0 +1,42 @@ +/*************************************************************************** +* Copyright (C) 2006 Jason Kivlighn ([email protected]) * +* * +* 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. * +***************************************************************************/ + +#ifndef WEIGHTINPUT_H +#define WEIGHTINPUT_H + +#include "widgets/amountunitinput.h" +#include "datablocks/unit.h" + +class RecipeDB; +class PrepMethodComboBox; +class Element; +class Weight; + +class WeightInput : public AmountUnitInput +{ +TQ_OBJECT + +public: + WeightInput( TQWidget *parent, RecipeDB *database, Unit::Type type = Unit::All, MixedNumber::Format f = MixedNumber::MixedNumberFormat ); + + Element prepMethod() const; + void setPrepMethod( const Element & ); + +public slots: + void emitValueChanged(); + +signals: + void valueChanged( const Weight & ); + +private: + PrepMethodComboBox *prepMethodBox; + + RecipeDB *m_database; +}; +#endif |