hongjli
2023-09-20 20d7889e98a86e9ed3dfe12b2a5ab6b3e43699c4
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
Quintiq file version 2.0
#parent: #root
MethodOverride InitConstraintsForPostponedSalesDemands (
  const CapacityPlanningSuboptimizer subopt,
  CapacityPlanningSuboptimizer_CapacityPlanningAlgorithm program,
  const LibOpt_Scope scope,
  Number optminpostponementperiod
) const
{
  TextBody:
  [*
    // Aggregated sales demands for higher product level
                  
    // Constraint: Sum of disaggregated and delayed disaggregated should equal or smaller than the aggregated quantity
    // DisaggregatedSalesDemandQty(childrenproducts) + DelayedDisaggregatedSalesDemandQty(childrenproducts) <= AggregatedSalesDemandQty forall Aggregated sdips
    sumofdisaggregatedsalesdemandconstrtype := typeof( MPSumOfDisaggregatedSalesDemandConstraint );
    scalefactor_disaggregatedsalesdemandqty_const := subopt.ScaleConstraintTerm( typeof( MPDisaggregatedSalesDemandQtyVariable ), sumofdisaggregatedsalesdemandconstrtype );
    
    constdisaggregated := program.SumOfDisaggregatedSalesDemandConstraints().New( this );
    constdisaggregated.Sense( '<=' );
    
    rhs := this.GetQuantityForOptimizer( scope );
    
    // Traverse the children ( Disaggregated Sales demands )
    traverse ( this, 
               DisaggregatedSalesDemandInPeriod, 
               dasdip, 
               not dasdip.IsPostponed() ) // note we only define DisaggregatedSalesDemandQtyVariables for unpostponed dasdip
    {
      // If the scope includes the PISPIP, then add its variable, otherwise subtract its fulfilled quantity from RHS
      if( scope.Contains( dasdip.AsPlanningBaseSalesDemandInPeriod().PISPIPInOptimizerRun() ) ) 
      {
        constdisaggregated.NewTerm( 1.0 * scalefactor_disaggregatedsalesdemandqty_const,
                                    program.DisaggregatedSalesDemandQtyVariables().Get( dasdip ) );
      }
      else
      {
        rhs := maxvalue( rhs - dasdip.FulfilledQuantity(), 0.0 );
      }
      
      // If the scope includes the PISP, then add the delayed variables, otherwise subtract the postponed quantity
      if( scope.Contains( dasdip.ProductInStockingPoint_MP().PISPInOptimizerRun() ) )
      {
        // add delayeddisaggregatedsalesdemand
        // add all delayed variables within the scope (outside of scope is not part of quantity computed above)
        dasdip.AddTermsForDelayedSalesDemandsForward( program, constdisaggregated, 1.0, scope, subopt, optminpostponementperiod );
      }
      else
      {
        postponedqty_inscope := sum( dasdip, 
                                     PostponedSalesDemand.astype( DisaggregatedSalesDemandInPeriod ), 
                                     postponed_dasdip, 
                                     scope.Contains( postponed_dasdip.ProductInStockingPointInPeriodPlanning().PISPIPInOptimizerRun() ), 
                                     postponed_dasdip.FulfilledQuantity() ); 
        rhs := maxvalue( rhs - postponedqty_inscope, 0.0 ); // take away from computed asdip quantity the postponed qty in scope
      }
    }
    constdisaggregated.RHSValue( subopt.ScaleConstraintRHS( sumofdisaggregatedsalesdemandconstrtype, rhs ) );
  *]
}