From 330ef75feffab3c58fe2fdda72f022d44eb4d18a Mon Sep 17 00:00:00 2001 From: Jean-Paul Smets <jp@nexedi.com> Date: Thu, 4 Mar 2004 10:31:07 +0000 Subject: [PATCH] Global Udate to Latest Coramy Optimizations (force in rules) git-svn-id: https://svn.erp5.org/repos/public/erp5/trunk@529 20353a03-c40f-0410-a6d1-a30d3c3de9de --- product/CMFCategory/CategoryTool.py | 15 + product/Coramy/Document/Modele.py | 1 + .../Coramy/skins/coramy_crm/person_view.form | 28 +- .../Delivery_rescueOrphanedMovement.py | 6 + .../coramy_erp5/SQLDict_readMessage.zsql | 2 +- .../coramy_erp5/SQLDict_readMessageList.zsql | 2 +- .../coramy_erp5/SQLDict_writeMessage.zsql | 2 +- .../assign_gestionaire_designe_roles.py | 4 +- .../coramy_mrp/Resource_zGetInventory.zsql | 4 +- .../Resource_zGetMovementHistoryList.zsql | 4 +- .../coramy_trade/SalesOrder_importEdi.py | 14 +- .../coramy_trade/default_pdf_template.pt | 32 +- .../sales_packing_list_aggregated_view.form | 89 +++++- .../sales_packing_list_container_view.form | 89 +++++- .../coramy_trade/sales_packing_list_print.pt | 2 +- .../coramy_trade/sales_packing_list_view.form | 64 +++- product/ERP5/Document/AppliedRule.py | 2 +- product/ERP5/Document/DeliveryRule.py | 8 +- product/ERP5/Document/Order.py | 12 +- product/ERP5/Document/OrderRule.py | 11 +- product/ERP5/Document/Rule.py | 4 +- product/ERP5/Document/SimulationMovement.py | 4 +- product/ERP5/Document/TransformationRule.py | 4 +- .../Document/TransformationSourcingRule.py | 4 +- product/ERP5/Document/ZeroStockRule.py | 2 +- product/ERP5/Tool/SimulationTool.py | 296 +++++++++--------- product/ERP5Type/Base.py | 1 + 27 files changed, 505 insertions(+), 201 deletions(-) diff --git a/product/CMFCategory/CategoryTool.py b/product/CMFCategory/CategoryTool.py index 176c698442..5912c7efab 100755 --- a/product/CMFCategory/CategoryTool.py +++ b/product/CMFCategory/CategoryTool.py @@ -903,6 +903,21 @@ class CategoryTool( UniqueObject, Folder, Base ): return 1 return 0 + security.declareProtected( Permissions.AccessContentsInformation, 'isAcquiredMemberOf' ) + def isAcquiredMemberOf(self, context, category): + """ + Tests if an object if member of a given category + Category is a string here. It could be more than a string (ex. an object) + + XXX Should include acquisition ? + """ + if getattr(aq_base(context), 'isCategory', 0): + return context.isAcquiredMemberOf(category) + for c in self._getAcquiredCategoryList(context): + if c.find(category) >= 0: + return 1 + return 0 + security.declareProtected( Permissions.AccessContentsInformation, 'getCategoryList' ) def getCategoryList(self, context): self._cleanupCategories(context) diff --git a/product/Coramy/Document/Modele.py b/product/Coramy/Document/Modele.py index db6b0dedd1..39d2c7da16 100755 --- a/product/Coramy/Document/Modele.py +++ b/product/Coramy/Document/Modele.py @@ -178,3 +178,4 @@ un modele...""" #pass self.modele_compute_pri(batch_mode=1) + diff --git a/product/Coramy/skins/coramy_crm/person_view.form b/product/Coramy/skins/coramy_crm/person_view.form index 76d167d99a..4b3f132ea0 100755 --- a/product/Coramy/skins/coramy_crm/person_view.form +++ b/product/Coramy/skins/coramy_crm/person_view.form @@ -217,60 +217,60 @@ <title>Right</title> <fields> - <field><id>my_organisation_title</id> <type>RelationStringField</type> + <field><id>my_default_email_text</id> <type>EmailField</type> <values> <alternate_name></alternate_name> - <base_category>subordination</base_category> - <catalog_index>Title</catalog_index> <css_class></css_class> <default></default> - <default_module>organisation</default_module> <description></description> <display_maxwidth></display_maxwidth> - <display_width type="int">20</display_width> + <display_width type="int">30</display_width> <external_validator></external_validator> <extra></extra> <hidden type="int">0</hidden> - <jump_method>base_jump_relation</jump_method> <max_length></max_length> - <portal_type type="list">[('Organisation', 'Organisation')]</portal_type> <required type="int">0</required> - <title>Organisation</title> + <title>Email</title> <truncate type="int">0</truncate> - <update_method>base_update_relation</update_method> </values> <tales> + <default>here/getDefaultEmailText</default> </tales> <messages> <message name="external_validator_failed">The input failed the external validator.</message> <message name="required_not_found">Input is required but no input given.</message> <message name="too_long">Too much input was given.</message> + <message name="not_email">You did not enter an email address.</message> </messages> </field> - <field><id>my_default_email_text</id> <type>EmailField</type> + <field><id>my_organisation_title</id> <type>RelationStringField</type> <values> <alternate_name></alternate_name> + <base_category>subordination</base_category> + <catalog_index>Title</catalog_index> <css_class></css_class> <default></default> + <default_module>organisation</default_module> <description></description> <display_maxwidth></display_maxwidth> - <display_width type="int">30</display_width> + <display_width type="int">20</display_width> <external_validator></external_validator> <extra></extra> <hidden type="int">0</hidden> + <jump_method>base_jump_relation</jump_method> <max_length></max_length> + <portal_type type="list">[('Organisation', 'Organisation')]</portal_type> <required type="int">0</required> - <title>Email</title> + <title>Organisation</title> <truncate type="int">0</truncate> + <update_method>base_update_relation</update_method> </values> <tales> - <default>here/getDefaultEmailText</default> </tales> <messages> <message name="external_validator_failed">The input failed the external validator.</message> <message name="required_not_found">Input is required but no input given.</message> <message name="too_long">Too much input was given.</message> - <message name="not_email">You did not enter an email address.</message> </messages> </field> <field><id>my_default_telephone_text</id> <type>StringField</type> diff --git a/product/Coramy/skins/coramy_erp5/Delivery_rescueOrphanedMovement.py b/product/Coramy/skins/coramy_erp5/Delivery_rescueOrphanedMovement.py index 0b4aba4a53..4f627d5d9e 100755 --- a/product/Coramy/skins/coramy_erp5/Delivery_rescueOrphanedMovement.py +++ b/product/Coramy/skins/coramy_erp5/Delivery_rescueOrphanedMovement.py @@ -76,4 +76,10 @@ if fix: for delivery in delivery_list: print " New delivery %s for causality %s" % (delivery.getRelativeUrl(), ' '.join(delivery.getCausalityList())) +# Invoke delivery rule +if len(requires_delivery_rule) > 0: + if fix: + context.updateAppliedRule() + print "Building new delivery rule" + return printed diff --git a/product/Coramy/skins/coramy_erp5/SQLDict_readMessage.zsql b/product/Coramy/skins/coramy_erp5/SQLDict_readMessage.zsql index ce53035350..fd34f808c7 100755 --- a/product/Coramy/skins/coramy_erp5/SQLDict_readMessage.zsql +++ b/product/Coramy/skins/coramy_erp5/SQLDict_readMessage.zsql @@ -17,4 +17,4 @@ WHERE <dtml-if priority> AND priority = <dtml-sqlvar priority type="int"> </dtml-if> ORDER BY - priority, date \ No newline at end of file + priority, date diff --git a/product/Coramy/skins/coramy_erp5/SQLDict_readMessageList.zsql b/product/Coramy/skins/coramy_erp5/SQLDict_readMessageList.zsql index e4e3fefa5a..b198b085fb 100755 --- a/product/Coramy/skins/coramy_erp5/SQLDict_readMessageList.zsql +++ b/product/Coramy/skins/coramy_erp5/SQLDict_readMessageList.zsql @@ -22,4 +22,4 @@ WHERE GROUP BY path, method_id, processing_node, processing ORDER BY - priority, date \ No newline at end of file + priority, date diff --git a/product/Coramy/skins/coramy_erp5/SQLDict_writeMessage.zsql b/product/Coramy/skins/coramy_erp5/SQLDict_writeMessage.zsql index b7d026d911..70fbb74a37 100755 --- a/product/Coramy/skins/coramy_erp5/SQLDict_writeMessage.zsql +++ b/product/Coramy/skins/coramy_erp5/SQLDict_writeMessage.zsql @@ -19,4 +19,4 @@ SET processing_node = -1, processing = -1, priority = <dtml-sqlvar priority type="int">, - message = <dtml-sqlvar message type="string"> \ No newline at end of file + message = <dtml-sqlvar message type="string"> diff --git a/product/Coramy/skins/coramy_erp5/assign_gestionaire_designe_roles.py b/product/Coramy/skins/coramy_erp5/assign_gestionaire_designe_roles.py index a01a9c56e4..a246b44f6c 100755 --- a/product/Coramy/skins/coramy_erp5/assign_gestionaire_designe_roles.py +++ b/product/Coramy/skins/coramy_erp5/assign_gestionaire_designe_roles.py @@ -26,4 +26,6 @@ elif local_user in equipe2 : elif local_user in equipe3 : object.AssignLocalRole(user_list=equipe3,role_list=['GestionaireDesigne',]) else : - object.AssignLocalRole(user_list=[local_user,],role_list=['GestionaireDesigne',]) + if user_name == "AnimatriceAppros" : + local_user = 'Michele_Kmiecik' + object.AssignLocalRole(user_list=[local_user,],role_list=['GestionaireDesigne',]) diff --git a/product/Coramy/skins/coramy_mrp/Resource_zGetInventory.zsql b/product/Coramy/skins/coramy_mrp/Resource_zGetInventory.zsql index 611cc67aee..a9b9a1d307 100755 --- a/product/Coramy/skins/coramy_mrp/Resource_zGetInventory.zsql +++ b/product/Coramy/skins/coramy_mrp/Resource_zGetInventory.zsql @@ -68,8 +68,8 @@ AND resource_membership.category_uid = resource_c.uid AND resource_membership.base_category_uid = resource_bc.uid AND resource_membership.uid = resource.uid </dtml-if><dtml-if omit_simulation>AND catalog.portal_type != "Simulation Movement" -</dtml-if><dtml-if omit_input>AND stock.quantity < 0 -</dtml-if><dtml-if omit_output>AND stock.quantity > 0 +</dtml-if><dtml-if omit_input>AND stock.quantity < 0 AND movement.source_uid <> movement.destination_uid +</dtml-if><dtml-if omit_output>AND stock.quantity > 0 AND movement.source_uid <> movement.destination_uid </dtml-if><dtml-if simulation_state>AND (<dtml-in simulation_state> catalog.simulation_state = <dtml-sqlvar sequence-item type="string"><dtml-if sequence-end><dtml-else> OR </dtml-if></dtml-in>) </dtml-if><dtml-if query>AND category.uid = node.uid AND <dtml-var query></dtml-if> \ No newline at end of file diff --git a/product/Coramy/skins/coramy_mrp/Resource_zGetMovementHistoryList.zsql b/product/Coramy/skins/coramy_mrp/Resource_zGetMovementHistoryList.zsql index 186fe2e01e..d427aa2136 100755 --- a/product/Coramy/skins/coramy_mrp/Resource_zGetMovementHistoryList.zsql +++ b/product/Coramy/skins/coramy_mrp/Resource_zGetMovementHistoryList.zsql @@ -67,8 +67,8 @@ AND movement.resource_uid = resource.uid </dtml-if><dtml-if to_date>AND movement.stop_date < <dtml-sqlvar to_date type="string"> </dtml-if><dtml-if "variation_text is not None">AND movement.variation_text = <dtml-sqlvar variation_text type="string"> </dtml-if><dtml-if omit_simulation>AND catalog.portal_type != "Simulation Movement" -</dtml-if><dtml-if omit_input>AND stock.quantity < 0 -</dtml-if><dtml-if omit_output>AND stock.quantity > 0 +</dtml-if><dtml-if omit_input>AND stock.quantity < 0 AND movement.source_uid <> movement.destination_uid +</dtml-if><dtml-if omit_output>AND stock.quantity > 0 AND movement.source_uid <> movement.destination_uid </dtml-if><dtml-if node_category>AND node_c.relative_url = <dtml-sqlvar node_category type="string"> AND node_membership.category_uid = node_c.uid AND node_membership.base_category_uid = node_bc.uid diff --git a/product/Coramy/skins/coramy_trade/SalesOrder_importEdi.py b/product/Coramy/skins/coramy_trade/SalesOrder_importEdi.py index 7a2fc43d57..784d2eac3d 100755 --- a/product/Coramy/skins/coramy_trade/SalesOrder_importEdi.py +++ b/product/Coramy/skins/coramy_trade/SalesOrder_importEdi.py @@ -4,7 +4,7 @@ ##bind namespace= ##bind script=script ##bind subpath=traverse_subpath -##parameters=import_file, segmentation_strategique='2003-2004', incoterm='DAF',delivery_mode='Transporteur', order_type='Approvisionnement', travel_duration=0, batch_mode=0,**kw +##parameters=import_file, segmentation_strategique='2003-2004', incoterm='DAF',delivery_mode='Transporteur', order_type='Approvisionnement', travel_duration=0, batch_mode=0,user_name='',**kw ##title= ## # import d'un fichier EDI @@ -70,9 +70,11 @@ sales_order.setDeliveryMode(delivery_mode) sales_order.setIncoterm(incoterm) sales_order.setOrderType(order_type) + # set the source administration -local_user = container.portal_membership.getAuthenticatedMember() -local_user_name = string.replace(local_user.getUserName(), ' ', '_') +#local_user = container.portal_membership.getAuthenticatedMember() +#local_user_name = string.replace(local_user.getUserName(), '_', ' ') +local_user_name = string.replace(user_name, '_', ' ') local_persons = sales_order.item_by_title_sql_search(title = local_user_name, portal_type = 'Person') if len(local_persons) > 0: sales_order.setSourceAdministration(local_persons[0].relative_url) @@ -272,7 +274,11 @@ sales_order.flushActivity(invoke=1) # try to apply a sale condition sales_order.sales_order_apply_condition(my_id, 1) - +# change the workflow to create the sales packing list +# give the role to user +sales_order.Item_doWorkflowTransition(workflow_action='user_order', workflow_id='order_workflow') +#sales_order.flushActivity(invoke=1) +sales_order.Item_doWorkflowTransition(workflow_action='user_confirm', workflow_id='order_workflow') # and this is the end .... if batch_mode: diff --git a/product/Coramy/skins/coramy_trade/default_pdf_template.pt b/product/Coramy/skins/coramy_trade/default_pdf_template.pt index 538125b955..7d7b8f1c93 100755 --- a/product/Coramy/skins/coramy_trade/default_pdf_template.pt +++ b/product/Coramy/skins/coramy_trade/default_pdf_template.pt @@ -526,6 +526,28 @@ <infostring align="left" x="1.1cm" y="3.2cm" size="5" font="Helvetica" color="(0,0,0)"> dues à cette date de règlement en principal, frais et taxes inclus, et au taux de l'intérêt légal majoré de 5 points, sans que cette pénalité puisse être en toute hypothèse inférieure à 1,5 fois le taux de l'intérêt légal. </infostring> + <infostring align="left" x="10cm" y="24.7cm" size="9" + font="Helvetica-Oblique" color="(0,0,0)"> + Répartition par tailles + </infostring> + <infostring align="left" x="15.4cm" y="24.7cm" size="9" + font="Helvetica-Oblique" color="(0,0,0)"> + Quantité + </infostring> + <infostring align="left" x="17.1cm" y="24.7cm" size="9" + font="Helvetica-Oblique" color="(0,0,0)"> + P.U.H.T. + </infostring> + <infostring align="left" x="18.8cm" y="24.7cm" size="9" + font="Helvetica-Oblique" color="(0,0,0)"> + Total H.T. + </infostring> + + <!-- Pied de facture --> + <rectangle x="1cm" y="5.0cm" width="19.5cm" height="0.8cm" + linewidth="0" fill="(0.75,0.75,0.75)"/> + <line x1="1cm" x2="20.5cm" y1="5.8cm" y2="5.8cm" width="1"/> + <line x1="1cm" x2="20.5cm" y1="5.0cm" y2="5.0cm" width="1"/> </static> @@ -608,7 +630,7 @@ <line x1="20.5cm" x2="20.5cm" y1="25cm" y2="4cm" width="1"/> <line x1="1cm" x2="20.5cm" y1="4cm" y2="4cm" width="1"/> - + <!-- Détail de la facture --> <rectangle x="1cm" y="24.6cm" width="19.5cm" height="0.4cm" linewidth="0" fill="(0.75,0.75,0.75)"/> @@ -696,12 +718,12 @@ <infostring align="center" x="3.8cm" y="4.5cm" size="9" font="Helvetica" color="(0,0,0)"> - + </infostring> <infostring align="left" x="6.5cm" y="4.5cm" size="9" font="Helvetica" color="(0,0,0)" tal:content="python: '%.2f' % income"> - + </infostring> <tal:block tal:condition="python: vad_recoverable"> <infostring align="center" x="9.7cm" y="4.5cm" size="9" @@ -712,7 +734,7 @@ </tal:block> <infostring align="left" x="10.7cm" y="4.5cm" size="9" font="Helvetica" color="(0,0,0)"> - + </infostring> <infostring align="center" x="13.8cm" y="4.5cm" size="10" font="Helvetica-Bold" color="(0,0,0)" @@ -722,7 +744,7 @@ <infostring align="center" x="16cm" y="4.5cm" size="10" font="Helvetica-Bold" color="(0,0,0)" tal:content="python: resource_id"> - resource_id XXX + resource_id XXX </infostring> <infostring align="left" x="18cm" y="4.5cm" size="10" font="Helvetica-Bold" color="(0,0,0)" diff --git a/product/Coramy/skins/coramy_trade/sales_packing_list_aggregated_view.form b/product/Coramy/skins/coramy_trade/sales_packing_list_aggregated_view.form index c101c8318b..3fccc19e08 100755 --- a/product/Coramy/skins/coramy_trade/sales_packing_list_aggregated_view.form +++ b/product/Coramy/skins/coramy_trade/sales_packing_list_aggregated_view.form @@ -140,6 +140,36 @@ <message name="unknown_selection">You selected an item that was not in the list.</message> </messages> </field> + <field><id>my_source_title</id> <type>RelationStringField</type> + <values> + <alternate_name></alternate_name> + <base_category>source</base_category> + <catalog_index>Title</catalog_index> + <css_class></css_class> + <default></default> + <default_module>organisation</default_module> + <description></description> + <display_maxwidth></display_maxwidth> + <display_width type="int">20</display_width> + <external_validator></external_validator> + <extra></extra> + <hidden type="int">0</hidden> + <jump_method>base_jump_relation</jump_method> + <max_length></max_length> + <portal_type type="list">[('Organisation', 'Organisation'), ('Category', 'Category'), ('Person', 'Person')]</portal_type> + <required type="int">0</required> + <title>Expédié par</title> + <truncate type="int">0</truncate> + <update_method>base_update_relation</update_method> + </values> + <tales> + </tales> + <messages> + <message name="external_validator_failed">The input failed the external validator.</message> + <message name="required_not_found">Input is required but no input given.</message> + <message name="too_long">Too much input was given.</message> + </messages> + </field> </fields> </group> <group> @@ -231,6 +261,61 @@ <message name="too_long">Too much input was given.</message> </messages> </field> + <field><id>destination_address</id> <type>StringField</type> + <values> + <alternate_name></alternate_name> + <css_class></css_class> + <default></default> + <description></description> + <display_maxwidth></display_maxwidth> + <display_width type="int">20</display_width> + <external_validator></external_validator> + <extra></extra> + <hidden type="int">1</hidden> + <max_length></max_length> + <required type="int">0</required> + <title>Lieu livraison : inconnu</title> + <truncate type="int">0</truncate> + </values> + <tales> + <title>python: here.getDestinationValue(portal_type=['Organisation']).Organisation_getOneLineAddress()</title> + </tales> + <messages> + <message name="external_validator_failed">The input failed the external validator.</message> + <message name="required_not_found">Input is required but no input given.</message> + <message name="too_long">Too much input was given.</message> + </messages> + </field> + <field><id>my_destination_section_title</id> <type>RelationStringField</type> + <values> + <alternate_name></alternate_name> + <base_category>destination_section</base_category> + <catalog_index>Title</catalog_index> + <css_class></css_class> + <default></default> + <default_module>organisation</default_module> + <description></description> + <display_maxwidth></display_maxwidth> + <display_width type="int">20</display_width> + <external_validator></external_validator> + <extra></extra> + <hidden type="int">0</hidden> + <jump_method>base_jump_relation</jump_method> + <max_length></max_length> + <portal_type type="list">[('Organisation', 'Organisation'), ('Category', 'Category'), ('Person', 'Person')]</portal_type> + <required type="int">0</required> + <title>Pour le compte de</title> + <truncate type="int">0</truncate> + <update_method>base_update_relation</update_method> + </values> + <tales> + </tales> + <messages> + <message name="external_validator_failed">The input failed the external validator.</message> + <message name="required_not_found">Input is required but no input given.</message> + <message name="too_long">Too much input was given.</message> + </messages> + </field> <field><id>my_start_date</id> <type>DateTimeField</type> <values> <alternate_name></alternate_name> @@ -248,7 +333,7 @@ <title>Date d'expédition prévue</title> </values> <tales> - <title>python:'Date d'expédition prévue : %s' % here.getStartDate()</title> + <title>python:'Expédition prévue le : ' + here.getStartDate().strftime('%d/%m/%Y')</title> </tales> <messages> <message name="external_validator_failed">The input failed the external validator.</message> @@ -299,7 +384,7 @@ <title>Date de livraison prévue</title> </values> <tales> - <title>python:'Date de livraison prévue : %s' % here.getStopDate()</title> + <title>python:'Livraison prévue le : ' + here.getStopDate().strftime('%d/%m/%Y')</title> </tales> <messages> <message name="external_validator_failed">The input failed the external validator.</message> diff --git a/product/Coramy/skins/coramy_trade/sales_packing_list_container_view.form b/product/Coramy/skins/coramy_trade/sales_packing_list_container_view.form index e5585965e1..56c2714f31 100755 --- a/product/Coramy/skins/coramy_trade/sales_packing_list_container_view.form +++ b/product/Coramy/skins/coramy_trade/sales_packing_list_container_view.form @@ -140,6 +140,36 @@ <message name="unknown_selection">You selected an item that was not in the list.</message> </messages> </field> + <field><id>my_source_title</id> <type>RelationStringField</type> + <values> + <alternate_name></alternate_name> + <base_category>source</base_category> + <catalog_index>Title</catalog_index> + <css_class></css_class> + <default></default> + <default_module>organisation</default_module> + <description></description> + <display_maxwidth></display_maxwidth> + <display_width type="int">20</display_width> + <external_validator></external_validator> + <extra></extra> + <hidden type="int">0</hidden> + <jump_method>base_jump_relation</jump_method> + <max_length></max_length> + <portal_type type="list">[('Organisation', 'Organisation'), ('Category', 'Category'), ('Person', 'Person')]</portal_type> + <required type="int">0</required> + <title>Expédié par</title> + <truncate type="int">0</truncate> + <update_method>base_update_relation</update_method> + </values> + <tales> + </tales> + <messages> + <message name="external_validator_failed">The input failed the external validator.</message> + <message name="required_not_found">Input is required but no input given.</message> + <message name="too_long">Too much input was given.</message> + </messages> + </field> </fields> </group> <group> @@ -231,6 +261,61 @@ <message name="too_long">Too much input was given.</message> </messages> </field> + <field><id>destination_address</id> <type>StringField</type> + <values> + <alternate_name></alternate_name> + <css_class></css_class> + <default></default> + <description></description> + <display_maxwidth></display_maxwidth> + <display_width type="int">20</display_width> + <external_validator></external_validator> + <extra></extra> + <hidden type="int">1</hidden> + <max_length></max_length> + <required type="int">0</required> + <title>Lieu livraison : inconnu</title> + <truncate type="int">0</truncate> + </values> + <tales> + <title>python: here.getDestinationValue(portal_type=['Organisation']).Organisation_getOneLineAddress()</title> + </tales> + <messages> + <message name="external_validator_failed">The input failed the external validator.</message> + <message name="required_not_found">Input is required but no input given.</message> + <message name="too_long">Too much input was given.</message> + </messages> + </field> + <field><id>my_destination_section_title</id> <type>RelationStringField</type> + <values> + <alternate_name></alternate_name> + <base_category>destination_section</base_category> + <catalog_index>Title</catalog_index> + <css_class></css_class> + <default></default> + <default_module>organisation</default_module> + <description></description> + <display_maxwidth></display_maxwidth> + <display_width type="int">20</display_width> + <external_validator></external_validator> + <extra></extra> + <hidden type="int">0</hidden> + <jump_method>base_jump_relation</jump_method> + <max_length></max_length> + <portal_type type="list">[('Organisation', 'Organisation'), ('Category', 'Category'), ('Person', 'Person')]</portal_type> + <required type="int">0</required> + <title>Pour le compte de</title> + <truncate type="int">0</truncate> + <update_method>base_update_relation</update_method> + </values> + <tales> + </tales> + <messages> + <message name="external_validator_failed">The input failed the external validator.</message> + <message name="required_not_found">Input is required but no input given.</message> + <message name="too_long">Too much input was given.</message> + </messages> + </field> <field><id>my_start_date</id> <type>DateTimeField</type> <values> <alternate_name></alternate_name> @@ -248,7 +333,7 @@ <title>Date d'expédition prévue</title> </values> <tales> - <title>python:'Date d'expédition prévue : %s' % here.getStartDate()</title> + <title>python:'Expédition prévue le : ' + here.getStartDate().strftime('%d/%m/%Y')</title> </tales> <messages> <message name="external_validator_failed">The input failed the external validator.</message> @@ -299,7 +384,7 @@ <title>Date de livraison prévue</title> </values> <tales> - <title>python:'Date de livraison prévue : %s' % here.getStopDate()</title> + <title>python:'Livraison prévue le : ' + here.getStopDate().strftime('%d/%m/%Y')</title> </tales> <messages> <message name="external_validator_failed">The input failed the external validator.</message> diff --git a/product/Coramy/skins/coramy_trade/sales_packing_list_print.pt b/product/Coramy/skins/coramy_trade/sales_packing_list_print.pt index f54aa27139..038ab2075b 100755 --- a/product/Coramy/skins/coramy_trade/sales_packing_list_print.pt +++ b/product/Coramy/skins/coramy_trade/sales_packing_list_print.pt @@ -8,7 +8,7 @@ page_number python:[0]"> <TABLE tal:repeat="delivery delivery_list" width="660" height="955" cellpadding="0" cellspacing="0" tal:attributes="class python:here.PT_pageBreak(page_number)"> <TABLE border="0" width="660" height="900" cellpadding="0" cellspacing="0" - tal:define="order python:delivery.getCausalityValue(portal_type=['Sales Order']); + tal:define="order python:delivery.getCausalityValue(portal_type=['Sales Order','Sales Packing List']); destination python:delivery.getDestinationValue(portal_type=['Organisation']); delivery_line_list python:filter(lambda line_item: line_item.getTargetTotalQuantity() > 0, delivery.contentValues(filter={'portal_type': 'Sales Packing List Line'})); diff --git a/product/Coramy/skins/coramy_trade/sales_packing_list_view.form b/product/Coramy/skins/coramy_trade/sales_packing_list_view.form index 1ec05e614e..cbaa677233 100755 --- a/product/Coramy/skins/coramy_trade/sales_packing_list_view.form +++ b/product/Coramy/skins/coramy_trade/sales_packing_list_view.form @@ -140,6 +140,36 @@ <message name="unknown_selection">You selected an item that was not in the list.</message> </messages> </field> + <field><id>my_source_title</id> <type>RelationStringField</type> + <values> + <alternate_name></alternate_name> + <base_category>source</base_category> + <catalog_index>Title</catalog_index> + <css_class></css_class> + <default></default> + <default_module>organisation</default_module> + <description></description> + <display_maxwidth></display_maxwidth> + <display_width type="int">20</display_width> + <external_validator></external_validator> + <extra></extra> + <hidden type="int">0</hidden> + <jump_method>base_jump_relation</jump_method> + <max_length></max_length> + <portal_type type="list">[('Organisation', 'Organisation'), ('Category', 'Category'), ('Person', 'Person')]</portal_type> + <required type="int">0</required> + <title>Expédié par</title> + <truncate type="int">0</truncate> + <update_method>base_update_relation</update_method> + </values> + <tales> + </tales> + <messages> + <message name="external_validator_failed">The input failed the external validator.</message> + <message name="required_not_found">Input is required but no input given.</message> + <message name="too_long">Too much input was given.</message> + </messages> + </field> </fields> </group> <group> @@ -248,7 +278,37 @@ <truncate type="int">0</truncate> </values> <tales> - <title>python: 'Lieu livraison : %s %s' % (here.getDestinationValue(portal_type=['Organisation']).getDefaultAddress().getZipCode(),here.getDestinationValue(portal_type=['Organisation']).getDefaultAddress().getCity())</title> + <title>python: here.getDestinationValue(portal_type=['Organisation']).Organisation_getOneLineAddress()</title> + </tales> + <messages> + <message name="external_validator_failed">The input failed the external validator.</message> + <message name="required_not_found">Input is required but no input given.</message> + <message name="too_long">Too much input was given.</message> + </messages> + </field> + <field><id>my_destination_section_title</id> <type>RelationStringField</type> + <values> + <alternate_name></alternate_name> + <base_category>destination_section</base_category> + <catalog_index>Title</catalog_index> + <css_class></css_class> + <default></default> + <default_module>organisation</default_module> + <description></description> + <display_maxwidth></display_maxwidth> + <display_width type="int">20</display_width> + <external_validator></external_validator> + <extra></extra> + <hidden type="int">0</hidden> + <jump_method>base_jump_relation</jump_method> + <max_length></max_length> + <portal_type type="list">[('Organisation', 'Organisation'), ('Category', 'Category'), ('Person', 'Person')]</portal_type> + <required type="int">0</required> + <title>Pour le compte de</title> + <truncate type="int">0</truncate> + <update_method>base_update_relation</update_method> + </values> + <tales> </tales> <messages> <message name="external_validator_failed">The input failed the external validator.</message> @@ -587,4 +647,4 @@ </fields> </group> </groups> -</form> \ No newline at end of file +</form> diff --git a/product/ERP5/Document/AppliedRule.py b/product/ERP5/Document/AppliedRule.py index ed7969dc43..5c322d44c2 100755 --- a/product/ERP5/Document/AppliedRule.py +++ b/product/ERP5/Document/AppliedRule.py @@ -163,7 +163,7 @@ An ERP5 Rule...""" rule.reset(self) security.declareProtected(Permissions.ModifyPortalContent, 'expand') - def expand(self): + def expand(self, **kw): """ Expands the current movement downward. diff --git a/product/ERP5/Document/DeliveryRule.py b/product/ERP5/Document/DeliveryRule.py index 4769f07de4..edfe7a6607 100755 --- a/product/ERP5/Document/DeliveryRule.py +++ b/product/ERP5/Document/DeliveryRule.py @@ -120,7 +120,7 @@ An ERP5 Rule...""" # Simulation workflow security.declareProtected(Permissions.ModifyPortalContent, 'expand') - def expand(self, applied_rule): + def expand(self, applied_rule, **kw): """ Expands the current movement downwards. @@ -136,7 +136,9 @@ An ERP5 Rule...""" # Only expand if my_delivery is not None and state is not 'confirmed' if my_delivery is not None: - if my_delivery.getSimulationState() not in ('delivered', ): + #if my_delivery.getSimulationState() not in ('delivered', ): + # Even if delivered, we should always calculate consequences + if 1: # First, check each contained movement and make # a list of delivery ids which do not need to be copied # eventually delete movement which do not exist anylonger @@ -216,7 +218,7 @@ An ERP5 Rule...""" % delivery_line_object.absolute_url()) # Pass to base class - Rule.expand(self, applied_rule) + Rule.expand(self, applied_rule, **kw) security.declareProtected(Permissions.ModifyPortalContent, 'solve') def solve(self, applied_rule, solution_list): diff --git a/product/ERP5/Document/Order.py b/product/ERP5/Document/Order.py index f8296ab3ba..30892ecf19 100755 --- a/product/ERP5/Document/Order.py +++ b/product/ERP5/Document/Order.py @@ -236,20 +236,24 @@ An order...""" my_applied_rule_list = self.getCausalityRelatedValueList(portal_type='Applied Rule') if len(my_applied_rule_list) != 1: # XXX This is an error - return + raise CategoryError, "Order has no or too many order rule(s)" applied_rule = my_applied_rule_list[0].getObject() if applied_rule is None: # XXX This is an error - return + raise CategoryError, "Order has None order rule" # Make sure applied rule has been reindexed - applied_rule.flushActivity(invoke=1) # Make sure there are no more activities on this order related to expand - self.flushActivity(invoke=1, method_id='expand') # Make sure expand is finished + self.flushActivity(invoke=0, method_id='expand') # Make sure expand is finished + # We are expanding but are not allowed to if state wrong... + # (ex. confirmed) + applied_rule.expand(force = 1) # thus, we mist force expand of applied order rule + applied_rule.flushActivity(invoke=1) # Build delivery list on applied rule # Currently, we build it 'again' but we should actually only build # deliveries for orphaned movements if self.getPortalType() == 'Production Order' : delivery_list = self.ProductionOrder_buildDeliveryList() # Coramy specific moved to portal_simulation + #else: elif self.getPortalType() in ('Purchase Order', 'Sales Order') : delivery_list = self.order_create_packing_list() # Coramy specific should be moved to portal_simulation #self.informDeliveryList(delivery_list=delivery_list, comment=repr(delivery_list)) # XXX Not ready diff --git a/product/ERP5/Document/OrderRule.py b/product/ERP5/Document/OrderRule.py index ac4e693d0e..0c23ae4f6f 100755 --- a/product/ERP5/Document/OrderRule.py +++ b/product/ERP5/Document/OrderRule.py @@ -29,7 +29,7 @@ from AccessControl import ClassSecurityInfo from Products.ERP5Type import Permissions, PropertySheet, Constraint, Interface from Products.ERP5.Document.Rule import Rule -from Products.ERP5.ERP5Globals import movement_type_list, order_movement_type_list +from Products.ERP5.ERP5Globals import movement_type_list, order_movement_type_list, reserved_inventory_state_list, current_inventory_state_list from zLOG import LOG @@ -121,7 +121,7 @@ An ERP5 Rule...""" # Simulation workflow security.declareProtected(Permissions.ModifyPortalContent, 'expand') - def expand(self, applied_rule): + def expand(self, applied_rule, force=0, **kw): """ Expands the current movement downward. @@ -141,8 +141,9 @@ An ERP5 Rule...""" if my_order is not None: # Only expand order rule if order not yet confirmed (This is consistent # with the fact that once simulation is launched, we stick to it) - if my_order.getSimulationState() not in reserved_inventory_state_list and \ - my_order.getSimulationState() not in current_inventory_state_list: + if force or \ + (my_order.getSimulationState() not in reserved_inventory_state_list and \ + my_order.getSimulationState() not in current_inventory_state_list): # First, check each contained movement and make # a list of order ids which do not need to be copied # eventually delete movement which do not exist anylonger @@ -198,7 +199,7 @@ An ERP5 Rule...""" % order_line_object.absolute_url()) # Pass to base class - Rule.expand(self, applied_rule) + Rule.expand(self, applied_rule, force=force, **kw) security.declareProtected(Permissions.ModifyPortalContent, 'solve') def solve(self, applied_rule, solution_list): diff --git a/product/ERP5/Document/Rule.py b/product/ERP5/Document/Rule.py index 5d6e2782c1..683e9e6cec 100755 --- a/product/ERP5/Document/Rule.py +++ b/product/ERP5/Document/Rule.py @@ -167,7 +167,7 @@ An ERP5 Rule...""" """ security.declareProtected(Permissions.ModifyPortalContent, 'expand') - def expand(self, applied_rule): + def expand(self, applied_rule, **kw): """ Expands the current movement downward. @@ -177,7 +177,7 @@ An ERP5 Rule...""" is expanded. """ for o in applied_rule.objectValues(): - o.expand() + o.expand(**kw) security.declareProtected(Permissions.ModifyPortalContent, 'solve') def solve(self, applied_rule, solution_list): diff --git a/product/ERP5/Document/SimulationMovement.py b/product/ERP5/Document/SimulationMovement.py index 39966ef731..d9aab201ee 100755 --- a/product/ERP5/Document/SimulationMovement.py +++ b/product/ERP5/Document/SimulationMovement.py @@ -236,7 +236,7 @@ a service in a public administration).""" # Causality Workflow Methods security.declareProtected(Permissions.ModifyPortalContent, 'expand') - def expand(self): + def expand(self, **kw): """ -> new status : expanded @@ -252,7 +252,7 @@ a service in a public administration).""" if self.getCausalityState() is 'expanded': # Reexpand for my_applied_rule in self.objectValues(): - my_applied_rule.expand() + my_applied_rule.expand(**kw) else: portal_rules = getToolByName(self, 'portal_rules') # Parse each applied rule and test if it applied diff --git a/product/ERP5/Document/TransformationRule.py b/product/ERP5/Document/TransformationRule.py index 533dfab4c5..b7805bbc1b 100755 --- a/product/ERP5/Document/TransformationRule.py +++ b/product/ERP5/Document/TransformationRule.py @@ -166,7 +166,7 @@ An ERP5 Rule...""" # Simulation workflow security.declareProtected(Permissions.ModifyPortalContent, 'expand') - def expand(self, applied_rule): + def expand(self, applied_rule, **kw): """ Expands the current movement downward. @@ -289,7 +289,7 @@ An ERP5 Rule...""" applied_rule._delObject(movement.getId()) # XXXX Make sur this is not deleted if already in delivery # Pass to base class - Rule.expand(self, applied_rule) + Rule.expand(self, applied_rule, **kw) security.declareProtected(Permissions.ModifyPortalContent, 'solve') def solve(self, applied_rule, solution_list): diff --git a/product/ERP5/Document/TransformationSourcingRule.py b/product/ERP5/Document/TransformationSourcingRule.py index c0e0c975c8..c117ecf0af 100755 --- a/product/ERP5/Document/TransformationSourcingRule.py +++ b/product/ERP5/Document/TransformationSourcingRule.py @@ -153,7 +153,7 @@ An ERP5 Rule...""" """ security.declareProtected(Permissions.ModifyPortalContent, 'expand') - def expand(self, applied_rule): + def expand(self, applied_rule, **kw): """ Expands the current movement downward. @@ -219,7 +219,7 @@ An ERP5 Rule...""" my_context_movement.getVariationCategoryList()) # Create one submovement which sources the transformation - Rule.expand(self, applied_rule) + Rule.expand(self, applied_rule, **kw) security.declareProtected(Permissions.ModifyPortalContent, 'solve') def solve(self, applied_rule, solution_list): diff --git a/product/ERP5/Document/ZeroStockRule.py b/product/ERP5/Document/ZeroStockRule.py index a88c7ff1ab..a15b64f957 100755 --- a/product/ERP5/Document/ZeroStockRule.py +++ b/product/ERP5/Document/ZeroStockRule.py @@ -123,7 +123,7 @@ An ERP5 Rule...""" # Simulation workflow security.declareProtected(Permissions.ModifyPortalContent, 'expand') - def expand(self, applied_rule): + def expand(self, applied_rule, **kw): """ An applied rule can be expanded only if its parent movement is expanded. diff --git a/product/ERP5/Tool/SimulationTool.py b/product/ERP5/Tool/SimulationTool.py index 6696e40c8b..b661a927d8 100755 --- a/product/ERP5/Tool/SimulationTool.py +++ b/product/ERP5/Tool/SimulationTool.py @@ -505,10 +505,12 @@ class SimulationTool (Folder, UniqueObject): if movement_group is not None: for order_group in movement_group.group_list: # Order should never be None + LOG("buildDeliveryList", 0, str(order_group.__dict__)) if order_group.order is not None: order = self.portal_categories.resolveCategory(order_group.order) if order is not None: # define some variables + LOG("order", 0, str(order.__dict__)) if order.getPortalType() == 'Purchase Order' : delivery_module = order.getPortalObject().livraison_achat delivery_type = 'Purchase Packing List' @@ -519,139 +521,151 @@ class SimulationTool (Folder, UniqueObject): delivery_type = 'Sales Packing List' delivery_line_type = delivery_type + ' Line' delivery_cell_type = 'Delivery Cell' - else: - #LOG("ERP5 Simulation", 100, "None order makes no sense") + else : # should never be none + LOG("order is None", 0, str(order.__dict__)) return delivery_list + else: # order is None + order = None + # possible when we build deliveries for tranfer of property + delivery_module = self.getPortalObject().livraison_vente + delivery_type = 'Sales Packing List' + delivery_line_type = delivery_type + ' Line' + delivery_cell_type = 'Delivery Cell' + + for path_group in order_group.group_list : + # we create a new delivery for each DateGroup + + # if path is internal ??? + # JPS NEW + if path_group.source is None or path_group.destination is None: + # Production Path + LOG("Builder",0, "Strange Path %s " % path_group.source) + LOG("Builder",0, "Strange Path %s " % path_group.destination) + + if path_group.source is None or path_group.destination is None: + delivery_module = self.rapport_fabrication + delivery_type = 'Production Report' + delivery_line_type = 'Production Report Line' + delivery_cell_type = 'Production Report Cell' + elif path_group.destination.find('site/Stock_PF') >= 0 and \ + path_group.source.find('site/Piquage') >= 0: + delivery_module = self.livraison_fabrication + delivery_type = 'Production Packing List' + delivery_line_type = delivery_type + ' Line' + delivery_cell_type = 'Delivery Cell' + elif path_group.source.find('site/Stock_MP') >= 0 and \ + path_group.destination.find('site/Piquage') >= 0: + delivery_module = self.livraison_fabrication + delivery_type = 'Production Packing List' + delivery_line_type = delivery_type + ' Line' + delivery_cell_type = 'Delivery Cell' + + for date_group in path_group.group_list : + + # Create a new packing list + new_delivery_id = str(delivery_module.generateNewId()) + self.portal_types.constructContent(type_name = delivery_type, + container = delivery_module, + id = new_delivery_id, + target_start_date = date_group.start_date, + target_stop_date = date_group.stop_date, + start_date = date_group.start_date, + stop_date = date_group.stop_date, + source = path_group.source, + destination = path_group.destination, + source_section = path_group.source_section, + destination_section = path_group.destination_section + ) + delivery = delivery_module[new_delivery_id] + if order is not None : + delivery.edit(title = order.getTitle(), + causality_value = order, + incoterm = order.getIncoterm(), + delivery_mode = order.getDeliveryMode() + ) + # the new delivery is added to the delivery_list + delivery_list.append(delivery) + # LOG('Livraison créée',0,str(delivery.getId())) + + # Create each delivery_line in the new delivery + + for resource_group in date_group.group_list : + if delivery_type == 'Production Report': + if resource_group.resource.find('operation') == 0: + delivery_line_type = 'Production Report Operation' + else: + delivery_line_type = 'Production Report Component' + + new_delivery_line_id = str(delivery.generateNewId()) + self.portal_types.constructContent(type_name = delivery_line_type, + container = delivery, + id = new_delivery_line_id, + resource = resource_group.resource, + ) + delivery_line = delivery[new_delivery_line_id] + #LOG('Ligne créée',0,str(delivery_line.getId())+' '+str(delivery_line.getResource())) + + line_variation_category_list = [] + line_variation_base_category_dict = {} + + # compute line_variation_base_category_list and + # line_variation_category_list for new delivery_line + for variant_group in resource_group.group_list : + for variation_item in variant_group.category_list : + if not variation_item in line_variation_category_list : + line_variation_category_list.append(variation_item) + variation_base_category_items = variation_item.split('/') + if len(variation_base_category_items) > 0 : + line_variation_base_category_dict[variation_base_category_items[0]] = 1 + + # update variation_base_category_list and line_variation_category_list for delivery_line + line_variation_base_category_list = line_variation_base_category_dict.keys() + delivery_line._setVariationBaseCategoryList(line_variation_base_category_list) + delivery_line.setVariationCategoryList(line_variation_category_list) + + # IMPORTANT : delivery cells are automatically created during setVariationCategoryList + + # update target_quantity for each delivery_cell + for variant_group in resource_group.group_list : + #LOG('Variant_group examin?,0,str(variant_group.category_list)) + object_to_update = None + # if there is no variation of the resource, update delivery_line with quantities and price + if len(variant_group.category_list) == 0 : + object_to_update = delivery_line + # else find which delivery_cell is represented by variant_group + else : + categories_identity = 0 + #LOG('Before Check cell',0,str(delivery_cell_type)) + #LOG('Before Check cell',0,str(delivery_line.contentValues())) + for delivery_cell in delivery_line.contentValues( + filter={'portal_type':delivery_cell_type}) : + #LOG('Check cell',0,str(delivery_cell)) + if len(variant_group.category_list) == len(delivery_cell.getVariationCategoryList()) : + #LOG('Parse category',0,str(delivery_cell.getVariationCategoryList())) + for category in delivery_cell.getVariationCategoryList() : + if not category in variant_group.category_list : + #LOG('Not found category',0,str(category)) + break + else : + categories_identity = 1 + + if categories_identity : + object_to_update = delivery_cell + break - for path_group in order_group.group_list : - # we create a new delivery for each DateGroup - - # if path is internal ??? - # JPS NEW - if path_group.source is None or path_group.destination is None: - # Production Path - #LOG("Builder",0, "Strange Path %s " % path_group.source) - #LOG("Builder",0, "Strange Path %s " % path_group.destination) - delivery_module = self.rapport_fabrication - delivery_type = 'Production Report' - delivery_line_type = 'Production Report Line' - delivery_cell_type = 'Production Report Cell' - elif path_group.destination.find('site/Stock_PF') >= 0 and \ - path_group.source.find('site/Piquage') >= 0: - delivery_module = self.livraison_fabrication - delivery_type = 'Production Packing List' - delivery_line_type = delivery_type + ' Line' - delivery_cell_type = 'Delivery Cell' - elif path_group.source.find('site/Stock_MP') >= 0 and \ - path_group.destination.find('site/Piquage') >= 0: - delivery_module = self.livraison_fabrication - delivery_type = 'Production Packing List' - delivery_line_type = delivery_type + ' Line' - delivery_cell_type = 'Delivery Cell' - - for date_group in path_group.group_list : - - # Create a new packing list - new_delivery_id = str(delivery_module.generateNewId()) - self.portal_types.constructContent(type_name = delivery_type, - container = delivery_module, - id = new_delivery_id, - title = order.getTitle(), - causality_value = order, - incoterm = order.getIncoterm(), - delivery_mode = order.getDeliveryMode(), - target_start_date = date_group.start_date, - target_stop_date = date_group.stop_date, - start_date = date_group.start_date, - stop_date = date_group.stop_date, - source = path_group.source, - destination = path_group.destination, - source_section = path_group.source_section, - destination_section = path_group.destination_section - ) - delivery = delivery_module[new_delivery_id] - # the new delivery is added to the delivery_list - delivery_list.append(delivery) - # LOG('Livraison créée',0,str(delivery.getId())) - - # Create each delivery_line in the new delivery - - for resource_group in date_group.group_list : - if delivery_type == 'Production Report': - if resource_group.resource.find('operation') == 0: - delivery_line_type = 'Production Report Operation' - else: - delivery_line_type = 'Production Report Component' - - new_delivery_line_id = str(delivery.generateNewId()) - self.portal_types.constructContent(type_name = delivery_line_type, - container = delivery, - id = new_delivery_line_id, - resource = resource_group.resource, - ) - delivery_line = delivery[new_delivery_line_id] - #LOG('Ligne créée',0,str(delivery_line.getId())+' '+str(delivery_line.getResource())) - - line_variation_category_list = [] - line_variation_base_category_dict = {} - - # compute line_variation_base_category_list and - # line_variation_category_list for new delivery_line - for variant_group in resource_group.group_list : - for variation_item in variant_group.category_list : - if not variation_item in line_variation_category_list : - line_variation_category_list.append(variation_item) - variation_base_category_items = variation_item.split('/') - if len(variation_base_category_items) > 0 : - line_variation_base_category_dict[variation_base_category_items[0]] = 1 - - # update variation_base_category_list and line_variation_category_list for delivery_line - line_variation_base_category_list = line_variation_base_category_dict.keys() - delivery_line._setVariationBaseCategoryList(line_variation_base_category_list) - delivery_line.setVariationCategoryList(line_variation_category_list) - - # IMPORTANT : delivery cells are automatically created during setVariationCategoryList - - # update target_quantity for each delivery_cell - for variant_group in resource_group.group_list : - #LOG('Variant_group examin',0,str(variant_group.category_list)) - object_to_update = None - # if there is no variation of the resource, update delivery_line with quantities and price - if len(variant_group.category_list) == 0 : - object_to_update = delivery_line - # else find which delivery_cell is represented by variant_group - else : - categories_identity = 0 - #LOG('Before Check cell',0,str(delivery_cell_type)) - #LOG('Before Check cell',0,str(delivery_line.contentValues())) - for delivery_cell in delivery_line.contentValues( - filter={'portal_type':delivery_cell_type}) : - #LOG('Check cell',0,str(delivery_cell)) - if len(variant_group.category_list) == len(delivery_cell.getVariationCategoryList()) : - #LOG('Parse category',0,str(delivery_cell.getVariationCategoryList())) - for category in delivery_cell.getVariationCategoryList() : - if not category in variant_group.category_list : - #LOG('Not found category',0,str(category)) - break - else : - categories_identity = 1 - - if categories_identity : - object_to_update = delivery_cell - break - - # compute target_quantity, quantity and price for delivery_cell or delivery_line and - # build relation between simulation_movement and delivery_cell or delivery_line - if object_to_update is not None : - cell_target_quantity = 0 - cell_total_price = 0 - for movement in variant_group.movement_list : - cell_target_quantity += movement.getNetConvertedTargetQuantity() - try: - cell_total_price += movement.getNetConvertedTargetQuantity()*movement.getPrice() # XXX WARNING - ADD PRICED QUANTITY - except: - cell_total_price = None - + # compute target_quantity, quantity and price for delivery_cell or delivery_line and + # build relation between simulation_movement and delivery_cell or delivery_line + if object_to_update is not None : + cell_target_quantity = 0 + cell_total_price = 0 + for movement in variant_group.movement_list : + cell_target_quantity += movement.getNetConvertedTargetQuantity() + try: + cell_total_price += movement.getNetConvertedTargetQuantity()*movement.getPrice() # XXX WARNING - ADD PRICED QUANTITY + except: + cell_total_price = None + + if movement.getPortalType() == 'Simulation Movement' : # update every simulation_movement # we set delivery_value and target dates and quantity movement._setDeliveryValue(object_to_update) @@ -665,16 +679,16 @@ class SimulationTool (Folder, UniqueObject): # We will reindex later reindexable_movement_list.append(movement) - if cell_target_quantity <> 0 and cell_total_price is not None: - average_price = cell_total_price/cell_target_quantity - else : - average_price = 0 - #LOG('object mis ?jour',0,str(object_to_update.getRelativeUrl())) - object_to_update._edit(target_quantity = cell_target_quantity, - quantity = cell_target_quantity, - price = average_price, - force_update = 1, - ) + if cell_target_quantity <> 0 and cell_total_price is not None: + average_price = cell_total_price/cell_target_quantity + else : + average_price = 0 + #LOG('object mis ?jour',0,str(object_to_update.getRelativeUrl())) + object_to_update._edit(target_quantity = cell_target_quantity, + quantity = cell_target_quantity, + price = average_price, + force_update = 1, + ) # If we reach this point, it means we could # create deliveries diff --git a/product/ERP5Type/Base.py b/product/ERP5Type/Base.py index 426081e26e..1854a8c3e5 100755 --- a/product/ERP5Type/Base.py +++ b/product/ERP5Type/Base.py @@ -495,6 +495,7 @@ class Base( CopyContainer, PortalContent, Base18, ActiveObject, ERP5PropertyMana security.declareProtected( Permissions.ModifyPortalContent, 'edit' ) def edit(self, REQUEST=None, force_update = 0, reindex_object=1, **kw): return self._edit(REQUEST=REQUEST, force_update=force_update, reindex_object=reindex_object, **kw) + edit = WorkflowMethod( edit ) # Accessing object property Ids -- 2.30.9