/******************************************************************************
 * Product: Adempiere ERP & CRM Smart Business Solution                       *
 * Copyright (C) 1999-2006 ComPiere, Inc. All Rights Reserved.                *
 * This program is free software; you can redistribute it and/or modify it    *
 * under the terms version 2 of the GNU General Public License as published   *
 * by the Free Software Foundation. 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.,    *
 * 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.                     *
 * For the text or an alternative of this public license, you may reach us    *
 * ComPiere, Inc., 2620 Augustine Dr. #245, Santa Clara, CA 95054, USA        *
 * or via info@compiere.org or http://www.compiere.org/license.html           *
 *****************************************************************************/
package org.compiere.model;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Timestamp;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.Properties;
import java.util.logging.Level;
import org.adempiere.exceptions.DBException;
import org.adempiere.exceptions.PeriodClosedException;
import org.compiere.util.CLogger;
import org.compiere.util.DB;
import org.compiere.util.DisplayType;
import org.compiere.util.Env;
import org.compiere.util.TimeUtil;
import org.compiere.util.Util;
import org.idempiere.cache.ImmutableIntPOCache;
import org.idempiere.cache.ImmutablePOSupport;
/**
 *  Calendar Period Model
 *
 *	@author Jorg Janke
 *	@version $Id: MPeriod.java,v 1.4 2006/07/30 00:51:05 jjanke Exp $
 *
 * @author Teo Sarca, SC ARHIPAC SERVICE SRL
 * 				
BF [ 1779438 ] Minor auto period control bug
 * 				BF [ 1893486 ] Auto Period Control return that period is always open
 *
 *  @author victor.perez@e-evolution.com, e-Evolution http://www.e-evolution.com
 * 			 FR [ 2520591 ] Support multiples calendar for Org 
 *			@see https://sourceforge.net/p/adempiere/feature-requests/631/ 
 */
public class MPeriod extends X_C_Period implements ImmutablePOSupport
{
	/**
	 * generated serial id
	 */
	private static final long serialVersionUID = -2625074973303489939L;
	/**
	 * Get Period from Cache (immutable)
	 * @param C_Period_ID id
	 * @return MPeriod
	 */
	public static MPeriod get (int C_Period_ID)
	{
		return get(Env.getCtx(), C_Period_ID);
	}
	
	/**
	 * Get Period from Cache (immutable)
	 * @param ctx context
	 * @param C_Period_ID id
	 * @return MPeriod
	 */
	public static MPeriod get (Properties ctx, int C_Period_ID)
	{
		if (C_Period_ID <= 0)
			return null;
		//
		Integer key = Integer.valueOf(C_Period_ID);
		MPeriod retValue = s_cache.get (ctx, key, e -> new MPeriod(ctx, e));
		if (retValue != null)
			return retValue;
		//
		retValue = new MPeriod (ctx, C_Period_ID, null);
		if (retValue.get_ID () == C_Period_ID)
		{
			s_cache.put (key, retValue, e -> new MPeriod(Env.getCtx(), e));
			return retValue;
		}
		return null;
	} 	//	get
	/**
	 * 	Find standard Period of DateAcct based on Client Calendar
	 *	@param ctx context
	 *	@param DateAcct date
	 *	@return active Period or null
	 *  @deprecated
	 */
	@Deprecated
	public static MPeriod get (Properties ctx, Timestamp DateAcct)
	{	
		return get(ctx, DateAcct, 0, null);
	}	//	get
	
	/**
	 * Find standard Period of DateAcct based on Client Calendar
	 * @param ctx context
	 * @param DateAcct date
	 * @param AD_Org_ID Organization
	 * @return active Period or null
	 */
	public static MPeriod get (Properties ctx, Timestamp DateAcct, int AD_Org_ID, String trxName)
	{
		
		if (DateAcct == null)
			return null;
		
        int C_Calendar_ID = getC_Calendar_ID(ctx,AD_Org_ID);
        
        return findByCalendar(ctx, DateAcct, C_Calendar_ID, trxName);
	}	//	get
	@Deprecated
	public static MPeriod get (Properties ctx, Timestamp DateAcct, int AD_Org_ID)
	{
		return get(ctx, DateAcct, AD_Org_ID, null);
	}
	/**
	 * Get updateable copy of MPeriod from cache
	 * @param ctx
	 * @param C_Period_ID
	 * @param trxName
	 * @return MPeriod
	 */
	public static MPeriod getCopy(Properties ctx, int C_Period_ID, String trxName)
	{
		MPeriod period = get(C_Period_ID);
		if (period != null)
			period = new MPeriod(ctx, period, trxName);
		return period;
	}
	/**
	 * @param ctx
	 * @param DateAcct
	 * @param C_Calendar_ID
	 * @return MPeriod
	 * @deprecated
	 */
	@Deprecated
	public static MPeriod findByCalendar(Properties ctx, Timestamp DateAcct, int C_Calendar_ID) {
		return findByCalendar(ctx, DateAcct, C_Calendar_ID, null);
	}
		
	/**
	 * Find standard Period of DateAcct 
	 * @param ctx
	 * @param DateAcct
	 * @param C_Calendar_ID
	 * @param trxName
	 * @return MPeriod or null
	 */
	public static MPeriod findByCalendar(Properties ctx, Timestamp DateAcct, int C_Calendar_ID, String trxName) {
		
		int AD_Client_ID = Env.getAD_Client_ID(ctx);
		//	Search in Cache first
		MPeriod[] it = s_cache.values().toArray(new MPeriod[0]);
		for (MPeriod period : it)
		{
			if (period.getC_Calendar_ID() == C_Calendar_ID && period.isStandardPeriod() && period.isInPeriod(DateAcct) 
					&& period.getAD_Client_ID() == AD_Client_ID)  // globalqss - CarlosRuiz - Fix [ 1820810 ] Wrong Period Assigned to Fact_Acct
				return period;
		}
		
		//	Get it from DB
	    MPeriod retValue = null;
		String sql = "SELECT * "
			+ "FROM C_Period "
			+ "WHERE C_Year_ID IN "
				+ "(SELECT C_Year_ID FROM C_Year WHERE C_Calendar_ID= ?)"
			+ " AND ? BETWEEN TRUNC(StartDate) AND TRUNC(EndDate)"
			+ " AND IsActive=? AND PeriodType=?";
        
		PreparedStatement pstmt = null;
		ResultSet rs = null;
		try
		{
			pstmt = DB.prepareStatement(sql, trxName);
            pstmt.setInt (1, C_Calendar_ID);
            pstmt.setTimestamp (2, TimeUtil.getDay(DateAcct));
            pstmt.setString(3, "Y");
            pstmt.setString(4, "S");
			rs = pstmt.executeQuery();
			while (rs.next())
			{
				MPeriod period = new MPeriod(ctx, rs, trxName);
				Integer key = Integer.valueOf(period.getC_Period_ID());
				s_cache.put (key, period);
				if (period.isStandardPeriod())
					retValue = period;
			}
		}
		catch (SQLException e)
		{
			s_log.log(Level.SEVERE, "DateAcct=" + DateAcct, e);
		}
		finally
		{
			DB.close(rs, pstmt);
			rs = null; pstmt = null;
		}
		if (retValue == null)
			if (s_log.isLoggable(Level.INFO)) s_log.info("No Standard Period for " + DateAcct 
				+ " (AD_Client_ID=" + AD_Client_ID + ")");
		return retValue;
	}
	/**
	 * 	Find valid standard Period of DateAcct based on Client Calendar
	 *	@param ctx context
	 *	@param DateAcct date
	 *	@return C_Period_ID or 0
	 *  @deprecated
	 */
	@Deprecated
	public static int getC_Period_ID (Properties ctx, Timestamp DateAcct)
	{
		MPeriod period = get (ctx, DateAcct, 0, null);
		if (period == null)
			return 0;
		return period.getC_Period_ID();
	}	//	getC_Period_ID
	
	/**
	 * 	Find valid standard Period of DateAcct based on Client Calendar
	 *	@param ctx context
	 *  @param DateAcct date
	 *  @param AD_Org_ID Organization
	 *	@return C_Period_ID or 0
	 */
	public static int getC_Period_ID (Properties ctx, Timestamp DateAcct, int AD_Org_ID)
	{
		MPeriod period = get (ctx, DateAcct, AD_Org_ID, null);
		if (period == null)
			return 0;
		return period.getC_Period_ID();
	}	//	getC_Period_ID
	/**
	 * 	Is standard Period Open for Document Base Type
	 *	@param ctx context
	 *	@param DateAcct date
	 *	@param DocBaseType base type
	 *	@return true if open
	 *  @deprecated
	 */
	@Deprecated
	public static boolean isOpen (Properties ctx, Timestamp DateAcct, String DocBaseType)
	{
		return isOpen(ctx, DateAcct,DocBaseType, 0 );
	}	//	isOpen
	
	/**
	 * 	Is standard Period Open for Document Base Type
	 *	@param ctx context
	 *  @param DateAcct date
	 *  @param DocBaseType base type (X_C_DocType.DOCBASETYPE_*)
	 *  @param AD_Org_ID Organization
	 *	@return true if open
	 */
	public static boolean isOpen(Properties ctx, Timestamp DateAcct, String DocBaseType, int AD_Org_ID)
	{
		return isOpen(ctx, DateAcct, DocBaseType, AD_Org_ID, false);
	}
	/**
	 * 	Is standard Period Open for Document Base Type
	 *	@param ctx context
	 *  @param DateAcct date
	 *  @param DocBaseType base type (X_C_DocType.DOCBASETYPE_*)
	 *  @param AD_Org_ID Organization
	 *  @param forPosting - true to check if the period is open for posting, false is for DocAction
	 *	@return true if open
	 */
	public static boolean isOpen (Properties ctx, Timestamp DateAcct, String DocBaseType, int AD_Org_ID, boolean forPosting)
	{
		if (DateAcct == null)
		{
			s_log.warning("No DateAcct");
			return false;
		}
		if (DocBaseType == null)
		{
			s_log.warning("No DocBaseType");
			return false;
		}
		MPeriod period = MPeriod.get (ctx, DateAcct, AD_Org_ID, null);
		if (period == null)
		{
			s_log.warning("No Period for " + DateAcct + " (" + DocBaseType + ")");
			return false;
		}
		boolean open = period.isOpen(DocBaseType, DateAcct, forPosting);
		if (!open)
			s_log.warning(period.getName()
				+ ": Not open for " + DocBaseType + " (" + DateAcct + ")");
		return open;
	}	//	isOpen
	/**
	 * Is standard Period Open - based on tableID+recordID (for IDEMPIERE-2392)
	 * @param ctx context
	 * @param tableID
	 * @param recordID
	 * @param trxName
	 * @return true if open
	 */
	public static boolean isOpen(Properties ctx, int tableID, int recordID, String trxName) {
		return isOpen (ctx, tableID, recordID, trxName, false);
	}
	/**
	 * Is standard Period Open - based on tableID+recordID (for IDEMPIERE-2392)
	 * @param ctx context
	 * @param tableID
	 * @param recordID
	 * @param trxName
	 * @param forPosting - true to check if the period is open for posting, false is for DocAction
	 * @return true if open
	 */
	public static boolean isOpen (Properties ctx, int tableID, int recordID, String trxName, boolean forPosting) {
		MTable table = MTable.get(ctx, tableID);
		PO po = table.getPO(recordID, trxName);
		// obtain DateAcct
		int idxdate = -1;
		if (   tableID == MInventory.Table_ID
				|| tableID == MMovement.Table_ID
				|| tableID == MProduction.Table_ID
				|| tableID == MProjectIssue.Table_ID) {
			idxdate = po.get_ColumnIndex("MovementDate");
		} else if (   tableID == MRequisition.Table_ID) {
			idxdate = po.get_ColumnIndex("DateDoc");
		} else if (   tableID == MAllocationHdr.Table_ID
				|| tableID == MMatchInv.Table_ID
				|| tableID == MMatchPO.Table_ID) {
			idxdate = po.get_ColumnIndex("DateTrx");
		} else {
			idxdate = po.get_ColumnIndex("DateAcct");
		}
		if (idxdate < 0) {
			if (s_log.isLoggable(Level.INFO)) s_log.info("Could not find DateAcct for " + table.getTableName());
			return true;
		}
		Timestamp dateAcct = null;
		Object objts = po.get_Value(idxdate);
		if (objts != null && objts instanceof Timestamp) {
			dateAcct = (Timestamp) objts;
		} else {
			s_log.warning("Could not find DateAcct (null or not Timestamp) for " + table.getTableName());
			return true;
		}
		// obtain DocBaseType
		String docBaseType = null;
		int idxdoctype = po.get_ColumnIndex("C_DocType_ID");
		if (idxdoctype < 0) {
			if (tableID == MInventory.Table_ID) {
				docBaseType = MDocType.DOCBASETYPE_MaterialPhysicalInventory;
			} else if (tableID == MRequisition.Table_ID) {
				docBaseType = MDocType.DOCBASETYPE_PurchaseRequisition;
			} else if (tableID == MAllocationHdr.Table_ID) {
				docBaseType = MDocType.DOCBASETYPE_PaymentAllocation;
			} else if (tableID == MMatchInv.Table_ID) {
				docBaseType = MDocType.DOCBASETYPE_MatchInvoice;
			} else if (tableID == MMatchPO.Table_ID) {
				docBaseType = MDocType.DOCBASETYPE_MatchPO;
			} else if (   tableID == MAssetAddition.Table_ID
					|| tableID == MAssetReval.Table_ID
					|| tableID == MAssetTransfer.Table_ID) {
				docBaseType = MDocType.DOCBASETYPE_GLJournal;
			} else if (   tableID == MAssetDisposed.Table_ID
					|| tableID == MDepreciationExp.Table_ID) {
				docBaseType = MDocType.DOCBASETYPE_GLDocument; // seems like a bug of fixed assets - must use GLJournal instead of GLDocument?
			} else if (tableID == MProjectIssue.Table_ID) {
				docBaseType = MDocType.DOCBASETYPE_ProjectIssue;
			} else {
				s_log.warning("Could not find C_DocType_ID for " + table.getTableName());
				return true;
			}
		} else {
			Integer doctypeID = null;
			Object objint = po.get_Value(idxdoctype);
			if (objint != null && objint instanceof Integer) {
				doctypeID = (Integer) objint;
			} else {
				s_log.warning("Could not find C_DocType_ID (null or not Integer) for " + table.getTableName());
				return true;
			}
			if (doctypeID == 0 && (tableID == MOrder.Table_ID || tableID == MInvoice.Table_ID)) {
				idxdoctype = po.get_ColumnIndex("C_DocTypeTarget_ID");
				objint = po.get_Value(idxdoctype);
				if (objint != null && objint instanceof Integer) {
					doctypeID = (Integer) objint;
				}
			}
			MDocType dt = MDocType.get(ctx, doctypeID);
			docBaseType = dt.getDocBaseType();
		}
		if (Util.isEmpty(docBaseType)) {
			s_log.warning("Could not find DocBaseType for " + table.getTableName());
			return true;
		}
		// obtain AD_Org_ID
		int orgID = 0;
		int idxorg = po.get_ColumnIndex("AD_Org_ID");
		if (idxorg < 0) {
			s_log.warning("Could not find AD_Org_ID for " + table.getTableName());
		} else {
			orgID = po.get_ValueAsInt(idxorg);
		}
		if (tableID == MJournal.Table_ID || tableID == MJournalBatch.Table_ID) {
			// special case for journal that has direct period
			int periodID = po.get_ValueAsInt("C_Period_ID");
			MPeriod period = MPeriod.get(ctx, periodID);
			boolean open = period.isOpen(docBaseType, dateAcct, forPosting);
			if (!open)
				s_log.warning(period.getName() + ": Not open for " + docBaseType + " (" + dateAcct + ")");
			return open;
		}
		return isOpen(ctx, dateAcct, docBaseType, orgID, forPosting);
	}	//	isOpen
	/**
	 * 	Find first Year Period of DateAcct based on Client Calendar
	 *	@param ctx context
	 *	@param DateAcct date
	 *	@return active first Period
	 *  @deprecated
	 */
	@Deprecated
	public static MPeriod getFirstInYear (Properties ctx, Timestamp DateAcct)
	{
		return getFirstInYear(ctx , DateAcct, 0);
	}	//	getFirstInYear
		
	/**
	 * 	Find first Year Period of DateAcct based on Client Calendar
	 *	@param ctx context
	 *  @param DateAcct date
	 *  @param AD_Org_ID organization
	 *	@return active first Period
	 */
	public static MPeriod getFirstInYear (Properties ctx, Timestamp DateAcct, int AD_Org_ID)
	{
		MPeriod retValue = null;
		int C_Calendar_ID = MPeriod.get(ctx, DateAcct, AD_Org_ID, null).getC_Calendar_ID();
        String sql = "SELECT * "
                    + "FROM C_Period "
                    + "WHERE C_Year_ID IN "
                    + "(SELECT p.C_Year_ID "
                    + "FROM C_Year y"
                    + " INNER JOIN C_Period p ON (y.C_Year_ID=p.C_Year_ID) "
                    + "WHERE y.C_Calendar_ID=?"
                    + "     AND ? BETWEEN StartDate AND EndDate)"
                    + " AND IsActive=? AND PeriodType=? "
                    + "ORDER BY StartDate";
        
		PreparedStatement pstmt = null;
		ResultSet rs = null;
		try
		{
			pstmt = DB.prepareStatement(sql, null);
			pstmt.setInt (1, C_Calendar_ID);
			pstmt.setTimestamp (2, DateAcct);
			pstmt.setString (3, "Y");
			pstmt.setString (4, "S");
			rs = pstmt.executeQuery();
			if (rs.next())	//	first only
				retValue = new MPeriod(ctx, rs, null);
		}
		catch (SQLException e)
		{
			s_log.log(Level.SEVERE, sql, e);
		}
		finally
		{
			DB.close(rs, pstmt);
			rs = null; pstmt = null;
		}
		return retValue;
	}	//	getFirstInYear
	/**	Cache							*/
	private static ImmutableIntPOCache s_cache = new ImmutableIntPOCache(Table_Name, 10);
	
	/**	Logger							*/
	private static CLogger			s_log = CLogger.getCLogger (MPeriod.class); 
	
	/** Calendar 					   */
	private int 					m_C_Calendar_ID = 0;
		
    /**
     * UUID based Constructor
     * @param ctx  Context
     * @param C_Period_UU  UUID key
     * @param trxName Transaction
     */
    public MPeriod(Properties ctx, String C_Period_UU, String trxName) {
        super(ctx, C_Period_UU, trxName);
		if (Util.isEmpty(C_Period_UU))
			setInitialDefaults();
    }
	/**
	 * 	Standard Constructor
	 *	@param ctx context
	 *	@param C_Period_ID id
	 *	@param trxName transaction
	 */
	public MPeriod (Properties ctx, int C_Period_ID, String trxName)
	{
		super (ctx, C_Period_ID, trxName);
		if (C_Period_ID == 0)
			setInitialDefaults();
	}	//	MPeriod
	/**
	 * Set the initial defaults for a new record
	 */
	private void setInitialDefaults() {
		setPeriodType (PERIODTYPE_StandardCalendarPeriod);
	}
	/**
	 * 	Load Constructor
	 *	@param ctx context
	 *	@param rs result set
	 *	@param trxName transaction
	 */
	public MPeriod (Properties ctx, ResultSet rs, String trxName)
	{
		super(ctx, rs, trxName);
	}	//	MPeriod
	/**
	 * 	Parent constructor
	 *	@param year year
	 *	@param PeriodNo no
	 *	@param name name
	 *	@param startDate start
	 *	@param endDate end
	 */
	public MPeriod (MYear year, int PeriodNo, String name, 
		Timestamp startDate,Timestamp endDate)
	{
		this (year.getCtx(), 0, year.get_TrxName());
		setClientOrg(year);
		setC_Year_ID(year.getC_Year_ID());
		setPeriodNo(PeriodNo);
		setName(name);
		setStartDate(startDate);
		setEndDate(endDate);
	}	//	MPeriod
	
	/**
	 * Copy constructor
	 * @param copy
	 */
	public MPeriod(MPeriod copy) 
	{
		this(Env.getCtx(), copy);
	}
	/**
	 * Copy constructor
	 * @param ctx
	 * @param copy
	 */
	public MPeriod(Properties ctx, MPeriod copy) 
	{
		this(ctx, copy, (String) null);
	}
	/**
	 * Copy constructor
	 * @param ctx
	 * @param copy
	 * @param trxName
	 */
	public MPeriod(Properties ctx, MPeriod copy, String trxName) 
	{
		this(ctx, 0, trxName);
		copyPO(copy);
		this.m_C_Calendar_ID = copy.m_C_Calendar_ID;
		this.m_controls = copy.m_controls != null ? Arrays.stream(copy.m_controls).map(e -> {return new MPeriodControl(ctx, e, trxName);}).toArray(MPeriodControl[]::new) : null;
	}
	/**	Period Controls			*/
	private MPeriodControl[] m_controls = null;
		
	/**
	 * 	Get Period Control
	 *	@param requery true to re-query from DB
	 *	@return period controls
	 */
	public MPeriodControl[] getPeriodControls (boolean requery)
	{
		if (m_controls != null && !requery)
			return m_controls;
		//
		ArrayList list = new ArrayList();
		String sql = "SELECT * FROM C_PeriodControl "
			+ "WHERE C_Period_ID=?";
		PreparedStatement pstmt = null;
		ResultSet rs = null;
		try
		{
			pstmt = DB.prepareStatement(sql, null);
			pstmt.setInt(1, getC_Period_ID());
			rs = pstmt.executeQuery();
			while (rs.next())
				list.add (new MPeriodControl (getCtx(), rs, null));
		}
		catch (Exception e)
		{
			log.log(Level.SEVERE, sql, e);
		}
		finally
		{
			DB.close(rs, pstmt);
			rs = null; pstmt = null;
		}
		//
		if (list.size() > 0 && is_Immutable())
			list.stream().forEach(e -> e.markImmutable());
		m_controls = new MPeriodControl[list.size ()];
		list.toArray (m_controls);
		return m_controls;
	}	//	getPeriodControls
	
	/**
	 * 	Get Period Control
	 *	@param DocBaseType Document Base Type (X_C_DocType.DOCBASETYPE_*)
	 *	@return period control or null
	 */
	public MPeriodControl getPeriodControl (String DocBaseType)
	{
		if (DocBaseType == null)
			return null;
		getPeriodControls(false);
		for (int i = 0; i < m_controls.length; i++)
		{
			if (DocBaseType.equals(m_controls[i].getDocBaseType()))
				return m_controls[i];
		}
		return null;
	}	//	getPeriodControl
	/**
	 * 	Is Date In this Period
	 *	@param date date
	 *	@return true if date is in this period
	 */
	public boolean isInPeriod (Timestamp date)
	{
		if (date == null)
			return false;
		Timestamp dateOnly = TimeUtil.getDay(date);
		Timestamp from = TimeUtil.getDay(getStartDate());
		if (dateOnly.before(from))
			return false;
		Timestamp to = TimeUtil.getDay(getEndDate());
		if (dateOnly.after(to))
			return false;
		return true;
	}	//	isInPeriod
	
	/**
	 * Is Period Open for Doc Base Type
	 * @param DocBaseType document base type
	 * @return true if open
	 * @deprecated since 3.3.1b; use {@link #isOpen(String, Timestamp)} instead
	 */
	@Deprecated
	public boolean isOpen (String DocBaseType)
	{
		return isOpen(DocBaseType, null);
	}
	
	/**
	 * Is Period Open for Doc Base Type
	 * @param DocBaseType document base type (X_C_DocType.DOCBASETYPE_*)
	 * @param dateAcct date;
	 * 			Applies only for "Auto Period Control":
	 * 			if not null, date should be in auto period range (today - OpenHistory, today+OpenHistory)
	 * 			if null, this period should be in auto period range
	 * @return true if open
	 * @since 3.3.1b
	 */
	public boolean isOpen (String DocBaseType, Timestamp dateAcct)
	{
		return isOpen(DocBaseType, dateAcct, false);
	}
	/**
	 * Is Period Open for Doc Base Type
	 * @param DocBaseType document base type
	 * @param dateAcct date;
	 * 			Applies only for "Auto Period Control":
	 * 			if not null, date should be in auto period range (today - OpenHistory, today+OpenHistory)
	 * 			if null, this period should be in auto period range
	 * @param forPosting - true to check if the period is open for posting, false is for DocAction
	 * @return true if open
	 */
	public boolean isOpen (String DocBaseType, Timestamp dateAcct, boolean forPosting)
	{
		if (!isActive())
		{
			s_log.warning("Period not active: " + getName());
			return false;
		}
		MAcctSchema as = MClient.get(getCtx(), getAD_Client_ID()).getAcctSchema();
		if (as != null && as.isAutoPeriodControl())
		{
			Timestamp today = TimeUtil.trunc(new Timestamp (System.currentTimeMillis()), TimeUtil.TRUNC_DAY);
			Timestamp first = TimeUtil.addDays(today, - as.getPeriod_OpenHistory()); 
			Timestamp last = TimeUtil.addDays(today, as.getPeriod_OpenFuture());
			Timestamp date1, date2;
			if (dateAcct != null) {
				date1 = TimeUtil.trunc(dateAcct, TimeUtil.TRUNC_DAY);
				date2 = date1;
			}
			else {
				date1 = getStartDate();
				date2 = getEndDate();
			}
			//
			if (date1.before(first))
			{
				log.warning ("Automatic Period Control:" + date1 + " before first day - " + first);
				return false;
			}
			if (date2.after(last))
			{
				log.warning ("Automatic Period Control:" + date2 + " after last day - " + last);
				return false;
			}
			//	We are OK
			if (isInPeriod(today) && as.getC_Period_ID() != getC_Period_ID())
			{
				as = new MAcctSchema(Env.getCtx(), as.getC_AcctSchema_ID(), null);
				if (as.getC_Period_ID() != getC_Period_ID())
				{
					as.setC_Period_ID(getC_Period_ID());
					as.saveEx();
				}
			}
			return true;
		}
		
		//	Standard Period Control
		if (DocBaseType == null)
		{
			log.warning(getName() + " - No DocBaseType");
			return false;
		}
		MPeriodControl pc = getPeriodControl (DocBaseType);
		if (pc == null)
		{
			log.warning(getName() + " - Period Control not found for " + DocBaseType);
			return false;
		}
		if (log.isLoggable(Level.FINE)) log.fine(getName() + ": " + DocBaseType);
		return pc.isOpen(forPosting);
	}	//	isOpen
	/**
	 * 	Standard Period
	 *	@return true if this is of standard calendar periods type (PERIODTYPE_StandardCalendarPeriod)
	 */
	public boolean isStandardPeriod()
	{
		return PERIODTYPE_StandardCalendarPeriod.equals(getPeriodType());
	}	//	isStandardPeriod
	
	
	/**
	 * 	Before Save.
	 * 	Truncate Dates
	 *	@param newRecord new
	 *	@return true
	 */
	@Override
	protected boolean beforeSave (boolean newRecord)
	{
		//	Truncate Dates
		Timestamp date = getStartDate(); 
		if (date != null)
			setStartDate(TimeUtil.getDay(date));
		else
			return false;
		//
		date = getEndDate();
		if (date != null)
			setEndDate(TimeUtil.getDay(date));
		else
			setEndDate(TimeUtil.getMonthLastDay(getStartDate()));
		
		if (getEndDate().before(getStartDate()))
		{
			SimpleDateFormat df = DisplayType.getDateFormat(DisplayType.Date);
			log.saveError("Error", df.format(getEndDate()) + " < " + df.format(getStartDate()));
			return false;
		}
		
		MYear year = new MYear(getCtx(), getC_Year_ID(), get_TrxName());
		
		Query query = MTable.get(getCtx(), "C_Period")
		.createQuery("C_Year_ID IN (SELECT y.C_Year_ID from C_Year y WHERE" +
				"                   y.C_Calendar_ID =?)" +
				" AND (? BETWEEN StartDate AND EndDate" +
				" OR ? BETWEEN StartDate AND EndDate)" +
				" AND PeriodType=?",get_TrxName());
		query.setParameters(year.getC_Calendar_ID(),
				getStartDate(), getEndDate(),
				getPeriodType());
		
		List periods = query.list();
		
		for ( int i=0; i < periods.size(); i++)
		{
			if ( periods.get(i).getC_Period_ID() != getC_Period_ID() )
			{
				log.saveError("Error", "Period overlaps with: "	+ periods.get(i).getName());
				return false;
			}
		}
		
		return true;
	}	//	beforeSave
	
	/**
	 * 	After Save
	 *	@param newRecord new
	 *	@param success success
	 *	@return success
	 */
	@Override
	protected boolean afterSave (boolean newRecord, boolean success)
	{
		if (!success)
			return success;
		if (newRecord)
		{
		//	SELECT Value FROM AD_Ref_List WHERE AD_Reference_ID=183
			MDocType[] types = MDocType.getOfClient(getCtx());
			int count = 0;
			ArrayList baseTypes = new ArrayList();
			for (int i = 0; i < types.length; i++)
			{
				MDocType type = types[i];
				String DocBaseType = type.getDocBaseType();
				if (baseTypes.contains(DocBaseType))
					continue;
				MPeriodControl pc = new MPeriodControl(this, DocBaseType);
				pc.saveEx();
				count++;
				baseTypes.add (DocBaseType);
			}
			if (log.isLoggable(Level.FINE)) log.fine("PeriodControl #" + count);
		}
		return success;
	}	//	afterSave
		
	/**
	 * 	String Representation
	 *	@return info
	 */
	@Override
	public String toString ()
	{
		StringBuilder sb = new StringBuilder ("MPeriod[");
		sb.append (get_ID())
			.append("-").append (getName())
			.append(", ").append(getStartDate()).append("-").append(getEndDate())
			.append ("]");
		return sb.toString ();
	}	//	toString
	
	/**
	 * Convenient method for testing if a period is open
	 * @param ctx
	 * @param dateAcct
	 * @param docBaseType
	 * @throws PeriodClosedException if period is closed
	 * @see #isOpen(Properties, Timestamp, String)
	 * @deprecated
	 */
	@Deprecated
	public static void testPeriodOpen(Properties ctx, Timestamp dateAcct, String docBaseType)
	throws PeriodClosedException 
	{
		if (!MPeriod.isOpen(ctx, dateAcct, docBaseType, 0)) {
			throw new PeriodClosedException(dateAcct, docBaseType);
		}
	}
	
	/**
	 * Convenient method for testing if a period is open
	 * @param ctx
	 * @param dateAcct
	 * @param docBaseType X_C_DocType.DOCBASETYPE_*
	 * @param AD_Org_ID Organization
	 * @throws PeriodClosedException if period is closed
	 * @see #isOpen(Properties, Timestamp, String, int)
	 */
	public static void testPeriodOpen(Properties ctx, Timestamp dateAcct, String docBaseType, int AD_Org_ID)
	throws PeriodClosedException 
	{
		if (!MPeriod.isOpen(ctx, dateAcct, docBaseType, AD_Org_ID)) {
			throw new PeriodClosedException(dateAcct, docBaseType);
		}
	}
	
	/**
	 * Convenient method for testing if a period is open
	 * @param ctx
	 * @param dateAcct
	 * @param C_DocType_ID
	 * @throws PeriodClosedException
	 * @see {@link #isOpen(Properties, Timestamp, String)}
     * @deprecated
	 */
	@Deprecated
	public static void testPeriodOpen(Properties ctx, Timestamp dateAcct, int C_DocType_ID)
	throws PeriodClosedException
	{
		MDocType dt = MDocType.get(ctx, C_DocType_ID);
		testPeriodOpen(ctx, dateAcct, dt.getDocBaseType(), 0);
	}
	
	/**
	 * Convenient method for testing if a period is open
	 * @param ctx
	 * @param dateAcct
	 * @param C_DocType_ID
	 * @param AD_Org_ID Organization
	 * @throws PeriodClosedException
	 * @see {@link #isOpen(Properties, Timestamp, String, int)}
	 */
	public static void testPeriodOpen(Properties ctx, Timestamp dateAcct, int C_DocType_ID, int AD_Org_ID)
	throws PeriodClosedException
	{
		MDocType dt = MDocType.get(ctx, C_DocType_ID);
		testPeriodOpen(ctx, dateAcct, dt.getDocBaseType(),  AD_Org_ID);
	}
	
	/**
	 *  Get Calendar of Period
	 *  @return C_Calendar_ID
	 */
	public int getC_Calendar_ID()
	{
		if (m_C_Calendar_ID == 0)
		{
			int calId = DB.getSQLValueEx(null, "SELECT C_Calendar_ID FROM C_Year WHERE C_Year_ID=?", getC_Year_ID());
			if (calId >= 0)
				m_C_Calendar_ID = calId;
			else
				log.severe("@NotFound@ C_Year_ID=" + getC_Year_ID());
		}
		return m_C_Calendar_ID;
	}   //  getC_Calendar_ID
    
	/**
	 * Get Calendar for Organization
	 * @param ctx Context
	 * @param AD_Org_ID Organization
	 * @return C_Calendar_ID
	 */
    public static int getC_Calendar_ID(Properties ctx,int AD_Org_ID)
    {	
        int C_Calendar_ID = 0;
        if (AD_Org_ID != 0)
        {
            MOrgInfo info = MOrgInfo.get(ctx, AD_Org_ID, null);
            C_Calendar_ID = info.getC_Calendar_ID();
        }
        
        if (C_Calendar_ID == 0)
        {
            MClientInfo cInfo = MClientInfo.get(ctx);
            C_Calendar_ID = cInfo.getC_Calendar_ID();
        }
        
      return C_Calendar_ID;
    }   //  getC_Calendar_ID
    
    @Override
	public MPeriod markImmutable() 
    {
    	if (is_Immutable())
    		return this;
    	
		makeImmutable();
		if (m_controls != null && m_controls.length > 0)
			Arrays.stream(m_controls).forEach(e -> e.markImmutable());
		return this;
	}
    /**
     * Is the period has un-posted documents
     * @return boolean - true if there is at least 1 un-posted document in this period
     */
    public boolean hasUnpostedDocs() {
    	return hasUnpostedDocs(0);
    }
    
    /**
     * Is this period has un-posted documents
     * @param periodControlID C_PeriodControl_ID. If > 0, check only documents with the period control's DocBaseType
     * @return boolean - true if there is at least 1 un-posted document in this period
     */
    public boolean hasUnpostedDocs(int periodControlID) {
		StringBuilder sql = new StringBuilder("SELECT 1 FROM RV_UnPosted up "
				+ "WHERE up.DocStatus IN('CO', 'CL', 'RE', 'VO') AND up.AD_Client_ID=? AND up.DateAcct BETWEEN ? AND ? ");
		sql.append(" AND AD_Org_ID IN (SELECT AD_Org_ID FROM AD_OrgInfo WHERE AD_Client_ID=? AND (C_Calendar_ID=?");
		if (getC_Calendar_ID() == MClientInfo.get().getC_Calendar_ID()) {
			sql.append(" OR C_Calendar_ID IS NULL");
		}
		sql.append(")) ");
		if (periodControlID > 0)
			sql.append(" AND up.DocBaseType = ? ");
		sql.append(" FETCH FIRST 1 ROWS ONLY");
    	PreparedStatement pstmt = null;
		ResultSet rs = null;
		try
		{
			pstmt = DB.prepareStatement(sql.toString(), get_TrxName());
			int idx = 1;
			pstmt.setInt(idx++, Env.getAD_Client_ID(Env.getCtx()));
			pstmt.setTimestamp(idx++, getStartDate());
			pstmt.setTimestamp(idx++, getEndDate());
			pstmt.setInt(idx++, Env.getAD_Client_ID(Env.getCtx()));
			pstmt.setInt(idx++, getC_Calendar_ID());
			if (periodControlID > 0) {
	    		MPeriodControl pc = new MPeriodControl(getCtx(), periodControlID, get_TrxName()); 
	    		pstmt.setString(idx++, pc.getDocBaseType());
	    	}
	    	
			rs = pstmt.executeQuery();
			
			if(rs.next()) {
				return true;
			}
		}
		catch (SQLException e)
		{
			throw new DBException(e, sql.toString());
		}
		finally
		{
			DB.close(rs, pstmt);
			rs = null; pstmt = null;
		}
    	return false;
    }
    
}	//	MPeriod