summaryrefslogtreecommitdiffstats
path: root/src/propertycalculator.cpp
diff options
context:
space:
mode:
authorMichele Calgaro <michele.calgaro@yahoo.it>2024-10-13 11:56:14 +0900
committerMichele Calgaro <michele.calgaro@yahoo.it>2024-10-29 21:58:42 +0900
commit2879ff70be9271550477982a1a6371714db38562 (patch)
treec2054149dba923ab080fe7093432c7663a990111 /src/propertycalculator.cpp
parent3eb38d2556f676d1027746f20bf12a1dd74451ef (diff)
downloadkrecipes-2879ff70be9271550477982a1a6371714db38562.tar.gz
krecipes-2879ff70be9271550477982a1a6371714db38562.zip
Rearrange folders structure to remove unnecessary 'krecipes' second level subfolder
Signed-off-by: Michele Calgaro <michele.calgaro@yahoo.it> (cherry picked from commit 0c8ed6c9a4000af8f48581a81c4b5c2f5b9fd502)
Diffstat (limited to 'src/propertycalculator.cpp')
-rw-r--r--src/propertycalculator.cpp131
1 files changed, 131 insertions, 0 deletions
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 *
+* ugarro@users.sourceforge.net *
+* *
+* 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 );
+ }
+ }
+ }
+}