From 777ec6b60b483e0a4916e8168a1fb363ed61c1df Mon Sep 17 00:00:00 2001 From: adambasha0 Date: Wed, 18 Dec 2024 16:02:28 +0000 Subject: [PATCH] feat: conversion feature rebased to v1.10 --- app/api/entities/reaction_material_entity.rb | 1 + app/api/helpers/report_helpers.rb | 2 + app/models/reactions_product_sample.rb | 1 + .../details/NumeralInputWithUnitsCompo.js | 2 +- .../details/reactions/schemeTab/Material.js | 81 +++++++++++++++---- .../reactions/schemeTab/MaterialGroup.js | 54 +++++++++++-- .../schemeTab/MaterialGroupContainer.js | 9 ++- .../schemeTab/ReactionDetailsScheme.js | 51 +++++++++++- .../src/components/common/ToggleButton.js | 18 ++++- app/packs/src/models/Sample.js | 1 + app/usecases/reactions/update_materials.rb | 4 +- ...dd_conversion_rate_to_reactions_samples.rb | 5 ++ db/schema.rb | 3 +- lib/import/import_collections.rb | 1 + lib/import/import_json.rb | 4 +- lib/reporter/docx/detail_reaction.rb | 3 +- .../components/common/ToggleButton.spec.js | 2 + 17 files changed, 205 insertions(+), 37 deletions(-) create mode 100644 db/migrate/20240917085816_add_conversion_rate_to_reactions_samples.rb diff --git a/app/api/entities/reaction_material_entity.rb b/app/api/entities/reaction_material_entity.rb index 7f9aa42c0d..f9c9d4d326 100644 --- a/app/api/entities/reaction_material_entity.rb +++ b/app/api/entities/reaction_material_entity.rb @@ -17,5 +17,6 @@ class ReactionMaterialEntity < ApplicationEntity expose! :waste expose! :gas_type expose! :gas_phase_data + expose! :conversion_rate end end diff --git a/app/api/helpers/report_helpers.rb b/app/api/helpers/report_helpers.rb index 1e24dc2a2a..63fd3c38dd 100644 --- a/app/api/helpers/report_helpers.rb +++ b/app/api/helpers/report_helpers.rb @@ -614,6 +614,7 @@ def build_sql_reaction_sample(columns, c_id, ids, checkedAll = false) # reactions_sample: equivalent: ['r_s.equivalent', '"r eq"', 10], reference: ['r_s.reference', '"r ref"', 10], + conversion_rate: ['r_s.conversion_rate', '"r conversion rate"', 10], }, analysis: { name: ['anac."name"', '"name"', 10], @@ -723,6 +724,7 @@ def force_molfile_selection equivalent reference type + conversion_rate ], sample: %i[ sample_svg_file diff --git a/app/models/reactions_product_sample.rb b/app/models/reactions_product_sample.rb index 062a19fcf0..05d58c4748 100644 --- a/app/models/reactions_product_sample.rb +++ b/app/models/reactions_product_sample.rb @@ -15,6 +15,7 @@ # show_label :boolean default(FALSE), not null # gas_type :integer default("off") # gas_phase_data :jsonb +# conversion_rate :float # # Indexes # diff --git a/app/packs/src/apps/mydb/elements/details/NumeralInputWithUnitsCompo.js b/app/packs/src/apps/mydb/elements/details/NumeralInputWithUnitsCompo.js index fe01aa05c9..da724b5c80 100644 --- a/app/packs/src/apps/mydb/elements/details/NumeralInputWithUnitsCompo.js +++ b/app/packs/src/apps/mydb/elements/details/NumeralInputWithUnitsCompo.js @@ -111,7 +111,7 @@ export default class NumeralInputWithUnitsCompo extends Component { togglePrefix(currentUnit) { const units = ['TON/h', 'TON/m', 'TON/s', '°C', '°F', 'K', 'h', 'm', 's']; - const excludedUnits = ['ppm', 'TON']; + const excludedUnits = ['ppm', 'TON', '%']; const { onMetricsChange, unit } = this.props; if (units.includes(currentUnit)) { // eslint-disable-next-line no-unused-expressions diff --git a/app/packs/src/apps/mydb/elements/details/reactions/schemeTab/Material.js b/app/packs/src/apps/mydb/elements/details/reactions/schemeTab/Material.js index 2c75247b45..a67f7a6923 100644 --- a/app/packs/src/apps/mydb/elements/details/reactions/schemeTab/Material.js +++ b/app/packs/src/apps/mydb/elements/details/reactions/schemeTab/Material.js @@ -123,6 +123,7 @@ class Material extends Component { this.gasFieldsUnitsChanged = this.gasFieldsUnitsChanged.bind(this); this.handleCoefficientChange = this.handleCoefficientChange.bind(this); this.debounceHandleAmountUnitChange = debounce(this.handleAmountUnitChange, 500); + this.yieldOrConversionRate = this.yieldOrConversionRate.bind(this); } handleMaterialClick(sample) { @@ -241,22 +242,45 @@ class Material extends Component { return result > 1 ? '100%' : `${(result * 100).toFixed(0)}%`; } - equivalentOrYield(material) { - const { reaction, materialGroup } = this.props; - if (materialGroup === 'products') { - const refMaterial = reaction.getReferenceMaterial(); - let calculateYield = material.equivalent; - if (material.gas_type === 'gas') { - calculateYield = this.recalculateYieldForGasProduct(material, reaction); - } else if (reaction.hasPolymers()) { - calculateYield = `${((material.equivalent || 0) * 100).toFixed(0)}%`; - } else if (refMaterial && (refMaterial.decoupled || material.decoupled)) { - calculateYield = 'n.a.'; - } else if (material.purity < 1 && material.equivalent > 1) { - calculateYield = `${((material.purity / 100 * (material.amount_g * 1000)) * 100).toFixed(1)}%`; - } else { - calculateYield = `${((material.equivalent <= 1 ? material.equivalent || 0 : 1) * 100).toFixed(0)}%`; - } + calculateYield(material, reaction) { + const refMaterial = reaction.getReferenceMaterial(); + let calculateYield = material.equivalent; + if (material.gas_type === 'gas') { + calculateYield = this.recalculateYieldForGasProduct(material, reaction); + } else if (reaction.hasPolymers()) { + calculateYield = `${((material.equivalent || 0) * 100).toFixed(0)}%`; + } else if (refMaterial && (refMaterial.decoupled || material.decoupled)) { + calculateYield = 'n.a.'; + } else if (material.purity < 1 && material.equivalent > 1) { + calculateYield = `${((material.purity / 100 * (material.amount_g * 1000)) * 100).toFixed(1)}%`; + } else { + calculateYield = `${((material.equivalent <= 1 ? material.equivalent || 0 : 1) * 100).toFixed(0)}%`; + } + return calculateYield; + } + + conversionRateField(material) { + const { reaction } = this.props; + const condition = material.conversion_rate / 100 > 1; + const allowedConversionRateValue = material.conversion_rate && condition + ? 100 : material.conversion_rate; + return ( +
+ this.handleConversionRateChange(e)} + size="sm" + /> +
+ ); + } + + yieldOrConversionRate(material) { + const { reaction, displayYieldField } = this.props; + if (displayYieldField === true || displayYieldField === null) { return (
); } + return this.conversionRateField(material); + } + + equivalentOrYield(material) { + const { materialGroup } = this.props; + if (materialGroup === 'products') { + return this.yieldOrConversionRate(material); + } return ( { const contents = []; let index = headIndex; @@ -36,6 +38,7 @@ const MaterialGroup = ({ dropMaterial={dropMaterial} dropSample={dropSample} lockEquivColumn={lockEquivColumn} + displayYieldField={displayYieldField} /> )); @@ -74,6 +77,8 @@ const MaterialGroup = ({ addDefaultSolvent={addDefaultSolvent} switchEquiv={switchEquiv} lockEquivColumn={lockEquivColumn} + displayYieldField={displayYieldField} + switchYield={switchYield} /> ); }; @@ -99,7 +104,7 @@ const SwitchEquivButton = (lockEquivColumn, switchEquiv) => { function GeneralMaterialGroup({ contents, materialGroup, showLoadingColumn, reaction, addDefaultSolvent, - switchEquiv, lockEquivColumn + switchEquiv, lockEquivColumn, displayYieldField, switchYield }) { const isReactants = materialGroup === 'reactants'; let headers = { @@ -151,9 +156,36 @@ function GeneralMaterialGroup({ ); } + const yieldConversionRateFields = () => { + const conversionText = 'Click to switch to conversion field.' + + ' The conversion will not be displayed as part of the reaction scheme'; + const yieldText = 'Click to switch to yield field.' + + ' The yield will be displayed as part of the reaction scheme'; + let conversionOrYield = displayYieldField; + if (displayYieldField || displayYieldField === null) { + conversionOrYield = true; + } + return ( +
+ +
+ ); + }; + if (materialGroup === 'products') { headers.group = 'Products'; - headers.eq = 'Yield'; + headers.eq = yieldConversionRateFields(); } const refTHead = (materialGroup !== 'products') ? headers.ref : null; @@ -213,7 +245,7 @@ function GeneralMaterialGroup({ {!isReactants && } {showLoadingColumn && !isReactants && {headers.loading}} {!isReactants && {headers.concn}} - {!isReactants && permitOn(reaction) && {headers.eq} {!isReactants && materialGroup !== 'products' && SwitchEquivButton(lockEquivColumn, switchEquiv)} } + {!isReactants && {headers.eq} {!isReactants && materialGroup !== 'products' && SwitchEquivButton(lockEquivColumn, switchEquiv)} } {contents.map(item => item)} @@ -310,7 +342,9 @@ MaterialGroup.propTypes = { dropMaterial: PropTypes.func.isRequired, dropSample: PropTypes.func.isRequired, switchEquiv: PropTypes.func.isRequired, - lockEquivColumn: PropTypes.bool + lockEquivColumn: PropTypes.bool, + displayYieldField: PropTypes.bool, + switchYield: PropTypes.func.isRequired }; GeneralMaterialGroup.propTypes = { @@ -320,7 +354,9 @@ GeneralMaterialGroup.propTypes = { addDefaultSolvent: PropTypes.func.isRequired, contents: PropTypes.arrayOf(PropTypes.shape).isRequired, switchEquiv: PropTypes.func.isRequired, - lockEquivColumn: PropTypes.bool + lockEquivColumn: PropTypes.bool, + displayYieldField: PropTypes.bool, + switchYield: PropTypes.func.isRequired }; SolventsMaterialGroup.propTypes = { @@ -332,12 +368,14 @@ SolventsMaterialGroup.propTypes = { MaterialGroup.defaultProps = { showLoadingColumn: false, - lockEquivColumn: false + lockEquivColumn: false, + displayYieldField: null }; GeneralMaterialGroup.defaultProps = { showLoadingColumn: false, - lockEquivColumn: false + lockEquivColumn: false, + displayYieldField: null }; diff --git a/app/packs/src/apps/mydb/elements/details/reactions/schemeTab/MaterialGroupContainer.js b/app/packs/src/apps/mydb/elements/details/reactions/schemeTab/MaterialGroupContainer.js index 6a5db73e3a..6af2658d64 100644 --- a/app/packs/src/apps/mydb/elements/details/reactions/schemeTab/MaterialGroupContainer.js +++ b/app/packs/src/apps/mydb/elements/details/reactions/schemeTab/MaterialGroupContainer.js @@ -55,7 +55,8 @@ class MaterialGroupContainer extends Component { const { materials, materialGroup, showLoadingColumn, headIndex, isOver, canDrop, connectDropTarget, - deleteMaterial, onChange, reaction, dropSample, dropMaterial, switchEquiv, lockEquivColumn + deleteMaterial, onChange, reaction, dropSample, dropMaterial, switchEquiv, lockEquivColumn, + displayYieldField, switchYield, } = this.props; const style = { padding: '2px 5px', @@ -82,6 +83,8 @@ class MaterialGroupContainer extends Component { headIndex={headIndex} switchEquiv={switchEquiv} lockEquivColumn={lockEquivColumn} + displayYieldField={displayYieldField} + switchYield={switchYield} /> , ); @@ -108,7 +111,9 @@ MaterialGroupContainer.propTypes = { canDrop: PropTypes.bool.isRequired, connectDropTarget: PropTypes.func.isRequired, switchEquiv: PropTypes.func, - lockEquivColumn: PropTypes.bool + lockEquivColumn: PropTypes.bool, + displayYieldField: PropTypes.bool.isRequired, + switchYield: PropTypes.func.isRequired, }; MaterialGroupContainer.defaultProps = { diff --git a/app/packs/src/apps/mydb/elements/details/reactions/schemeTab/ReactionDetailsScheme.js b/app/packs/src/apps/mydb/elements/details/reactions/schemeTab/ReactionDetailsScheme.js index 6f70c58ad8..ffc912f0d9 100644 --- a/app/packs/src/apps/mydb/elements/details/reactions/schemeTab/ReactionDetailsScheme.js +++ b/app/packs/src/apps/mydb/elements/details/reactions/schemeTab/ReactionDetailsScheme.js @@ -47,6 +47,7 @@ export default class ReactionDetailsScheme extends Component { reaction, lockEquivColumn: false, cCon: false, + displayYieldField: null, reactionDescTemplate: textTemplate.toJS(), open: true, }; @@ -63,6 +64,7 @@ export default class ReactionDetailsScheme extends Component { this.dropMaterial = this.dropMaterial.bind(this); this.dropSample = this.dropSample.bind(this); this.switchEquiv = this.switchEquiv.bind(this); + this.switchYield = this.switchYield.bind(this); this.handleOnConditionSelect = this.handleOnConditionSelect.bind(this); this.updateTextTemplates = this.updateTextTemplates.bind(this); this.reactionVesselSize = this.reactionVesselSize.bind(this); @@ -137,6 +139,11 @@ export default class ReactionDetailsScheme extends Component { this.setState({ lockEquivColumn: !lockEquivColumn }); } + switchYield = (shouldDisplayYield) => { + const { displayYieldField } = this.state; + this.setState({ displayYieldField: shouldDisplayYield ?? !displayYieldField }); + }; + handleOnConditionSelect(eventKey) { const { reaction } = this.props; const val = eventKey.value; @@ -363,6 +370,11 @@ export default class ReactionDetailsScheme extends Component { this.updatedReactionForGasFieldsUnitsChange(changeEvent) ); break; + case 'conversionRateChanged': + this.onReactionChange( + this.updatedReactionForConversionRateChange(changeEvent) + ); + break; default: break; } @@ -635,6 +647,22 @@ export default class ReactionDetailsScheme extends Component { ); } + updatedReactionForConversionRateChange(changeEvent) { + const { reaction } = this.props; + const { sampleID, conversionRate } = changeEvent; + const updatedSample = reaction.sampleById(sampleID); + + updatedSample.conversion_rate = conversionRate; + if (conversionRate / 100 > 1) { + NotificationActions.add({ + message: 'conversion rate cannot be more than 100%', + level: 'warning' + }); + } + + return this.updatedReactionWithSample(this.updatedSamplesForConversionRateChange.bind(this), updatedSample); + } + calculateEquivalent(refM, updatedSample) { if (!refM.contains_residues) { NotificationActions.add({ @@ -1031,6 +1059,15 @@ export default class ReactionDetailsScheme extends Component { return reaction; } + updatedSamplesForConversionRateChange(samples, updatedSample) { + return samples.map((sample) => { + if (sample.id === updatedSample.id) { + sample.conversion_rate = updatedSample.conversion_rate; + } + return sample; + }); + } + solventCollapseBtn() { const { open } = this.state; const arrow = open @@ -1138,7 +1175,8 @@ export default class ReactionDetailsScheme extends Component { const { reaction, lockEquivColumn, - reactionDescTemplate + reactionDescTemplate, + displayYieldField } = this.state; const minPadding = { padding: '1px 2px 2px 0px' }; if (reaction.editedSample !== undefined) { @@ -1176,6 +1214,15 @@ export default class ReactionDetailsScheme extends Component { reaction.markSampleAsReference(refM.id); } + if (displayYieldField === null) { + const allHaveNoConversion = reaction.products.every( + (material) => material.conversion_rate && material.conversion_rate !== 0 + ); + if (allHaveNoConversion) { + this.switchYield(!allHaveNoConversion); + } + } + const headReactants = reaction.starting_materials.length ?? 0; return (
@@ -1229,6 +1276,8 @@ export default class ReactionDetailsScheme extends Component { onChange={changeEvent => this.handleMaterialsChange(changeEvent)} switchEquiv={this.switchEquiv} lockEquivColumn={this.state.lockEquivColumn} + switchYield={this.switchYield} + displayYieldField={displayYieldField} headIndex={0} /> diff --git a/app/packs/src/components/common/ToggleButton.js b/app/packs/src/components/common/ToggleButton.js index ba3546b37c..b3c89cf293 100644 --- a/app/packs/src/components/common/ToggleButton.js +++ b/app/packs/src/components/common/ToggleButton.js @@ -1,10 +1,10 @@ -import React, { useState } from 'react'; +import React, { useState, useEffect } from 'react'; import { Button, OverlayTrigger, Tooltip } from 'react-bootstrap'; import PropTypes from 'prop-types'; export default function ToggleButton({ isToggledInitial, onToggle, onChange, onLabel, offLabel, - onColor, offColor, tooltipOn, tooltipOff + onColor, offColor, tooltipOn, tooltipOff, fontSize, fontWeight }) { const [isToggled, setIsToggled] = useState(isToggledInitial); @@ -15,6 +15,10 @@ export default function ToggleButton({ if (onChange) onChange(newToggledState); }; + useEffect(() => { + setIsToggled(isToggledInitial); + }, [isToggledInitial]); + const buttonColor = isToggled ? onColor : offColor; const toolTipMessage = isToggled ? tooltipOn : tooltipOff; @@ -22,11 +26,13 @@ export default function ToggleButton({ {toolTipMessage}}> ); @@ -42,6 +48,8 @@ ToggleButton.propTypes = { offColor: PropTypes.string, tooltipOn: PropTypes.string, tooltipOff: PropTypes.string, + fontWeight: PropTypes.string, + fontSize: PropTypes.string, }; ToggleButton.defaultProps = { @@ -53,4 +61,6 @@ ToggleButton.defaultProps = { offColor: '#d3d3d3', tooltipOn: 'Click to switch off', tooltipOff: 'Click to switch on', + fontWeight: 'normal', + fontSize: '1em', }; diff --git a/app/packs/src/models/Sample.js b/app/packs/src/models/Sample.js index 5e3947ec76..cc1ec53abb 100644 --- a/app/packs/src/models/Sample.js +++ b/app/packs/src/models/Sample.js @@ -1047,6 +1047,7 @@ export default class Sample extends Element { coefficient: this.coefficient, gas_type: this.gas_type || false, gas_phase_data: this.gas_phase_data, + conversion_rate: this.conversion_rate, }; _.merge(params, extra_params); return params; diff --git a/app/usecases/reactions/update_materials.rb b/app/usecases/reactions/update_materials.rb index 7badd390c2..b734b71e82 100644 --- a/app/usecases/reactions/update_materials.rb +++ b/app/usecases/reactions/update_materials.rb @@ -113,7 +113,7 @@ def create_new_sample(sample, fixed_label) :type, :molecule, :collection_id, :short_label, :waste, :show_label, :coefficient, :user_labels, :boiling_point_lowerbound, :boiling_point_upperbound, :melting_point_lowerbound, :melting_point_upperbound, :segments, :gas_type, - :gas_phase_data + :gas_phase_data, :conversion_rate ).merge(created_by: @current_user.id, boiling_point: rangebound(sample.boiling_point_lowerbound, sample.boiling_point_upperbound), melting_point: rangebound(sample.melting_point_lowerbound, sample.melting_point_upperbound)) @@ -215,6 +215,7 @@ def associate_sample_with_reaction(sample, modified_sample, material_group) type: reactions_sample_klass, gas_type: sample.gas_type, gas_phase_data: sample.gas_phase_data, + conversion_rate: sample.conversion_rate, ) # sample was moved to other materialgroup else @@ -230,6 +231,7 @@ def associate_sample_with_reaction(sample, modified_sample, material_group) type: reactions_sample_klass, gas_type: sample.gas_type, gas_phase_data: sample.gas_phase_data, + conversion_rate: sample.conversion_rate, ) end end diff --git a/db/migrate/20240917085816_add_conversion_rate_to_reactions_samples.rb b/db/migrate/20240917085816_add_conversion_rate_to_reactions_samples.rb new file mode 100644 index 0000000000..2118a61692 --- /dev/null +++ b/db/migrate/20240917085816_add_conversion_rate_to_reactions_samples.rb @@ -0,0 +1,5 @@ +class AddConversionRateToReactionsSamples < ActiveRecord::Migration[6.1] + def change + add_column :reactions_samples, :conversion_rate, :float, null: true + end +end diff --git a/db/schema.rb b/db/schema.rb index 3d7be91070..722bb60d23 100644 --- a/db/schema.rb +++ b/db/schema.rb @@ -10,7 +10,7 @@ # # It's strongly recommended that you check this file into your version control system. -ActiveRecord::Schema.define(version: 2024_08_06_082351) do +ActiveRecord::Schema.define(version: 2024_09_17_085816) do # These are extensions that must be enabled in order to support this database enable_extension "hstore" @@ -983,6 +983,7 @@ t.boolean "show_label", default: false, null: false t.integer "gas_type", default: 0 t.jsonb "gas_phase_data", default: {"time"=>{"unit"=>"h", "value"=>nil}, "temperature"=>{"unit"=>"°C", "value"=>nil}, "turnover_number"=>nil, "part_per_million"=>nil, "turnover_frequency"=>{"unit"=>"TON/h", "value"=>nil}} + t.float "conversion_rate" t.index ["reaction_id"], name: "index_reactions_samples_on_reaction_id" t.index ["sample_id"], name: "index_reactions_samples_on_sample_id" end diff --git a/lib/import/import_collections.rb b/lib/import/import_collections.rb index de3d257dbb..a326a858a0 100644 --- a/lib/import/import_collections.rb +++ b/lib/import/import_collections.rb @@ -388,6 +388,7 @@ def import_reactions_samples 'coefficient', 'gas_type', 'gas_phase_data', + 'conversion_rate', ).merge( reaction: @instances.fetch('Reaction').fetch(fields.fetch('reaction_id')), sample: @instances.fetch('Sample').fetch(fields.fetch('sample_id')), diff --git a/lib/import/import_json.rb b/lib/import/import_json.rb index a59314421a..53a8d9262d 100644 --- a/lib/import/import_json.rb +++ b/lib/import/import_json.rb @@ -82,6 +82,7 @@ class Import::ImportJson # 'r_uuid' => nil, # 'r_reference' => nil, # 'r_equivalent' => nil, + # 'r_conversion_rate' => nil, # ] # } # end @@ -374,7 +375,8 @@ def add_to_reaction(klass, el, new_el) if new_data && new_data[r_uuid] && new_data[r_uuid]['id'] @log['samples'][el_uuid][klass.name] = klass.create( sample_id: new_el.id, reaction_id: new_data[r_uuid]['id'], - reference: ref, equivalent: eq, position: el['r_position'] + reference: ref, equivalent: eq, position: el['r_position'], + conversion_rate: el['r_conversion_rate'] ) && '201' || '500' else @log['samples'][el_uuid][klass.name] = '404' diff --git a/lib/reporter/docx/detail_reaction.rb b/lib/reporter/docx/detail_reaction.rb index cbfda585a7..71be02eca3 100644 --- a/lib/reporter/docx/detail_reaction.rb +++ b/lib/reporter/docx/detail_reaction.rb @@ -361,7 +361,7 @@ def material_hash(material, is_product=false) mol: valid_digit(mmol, digit), mmol_unit: mmol_unit, equiv: valid_digit(s.equivalent, digit), - molecule_name_hash: s[:molecule_name_hash] + molecule_name_hash: s[:molecule_name_hash], } if is_product @@ -372,6 +372,7 @@ def material_hash(material, is_product=false) mol: valid_digit(mmol, digit), equiv: equiv, molecule_name_hash: s[:molecule_name_hash], + conversion_rate: s.conversion_rate, }) end diff --git a/spec/javascripts/packs/src/components/common/ToggleButton.spec.js b/spec/javascripts/packs/src/components/common/ToggleButton.spec.js index 5e2ffefc27..429ff5c112 100644 --- a/spec/javascripts/packs/src/components/common/ToggleButton.spec.js +++ b/spec/javascripts/packs/src/components/common/ToggleButton.spec.js @@ -27,6 +27,8 @@ describe('', () => { offColor="#d3d3d3" tooltipOn="Click to enable Default mode" tooltipOff="Click to enable Gas mode" + fontSize="1em" + fontWeight="normal" /> ); });