mirror of
https://github.com/YunoHost-Apps/noalyss_ynh.git
synced 2024-09-03 19:46:20 +02:00
3a905a4a87
Update files from sources with last update on noalyss.eu
2882 lines
101 KiB
PHP
2882 lines
101 KiB
PHP
<?php
|
|
|
|
/*
|
|
* This file is part of NOALYSS.
|
|
*
|
|
* NOALYSS is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* NOALYSS 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 NOALYSS; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
// Copyright Author Dany De Bontridder danydb@aevalys.eu
|
|
require_once NOALYSS_INCLUDE.'/lib/iselect.class.php';
|
|
require_once NOALYSS_INCLUDE.'/lib/icard.class.php';
|
|
require_once NOALYSS_INCLUDE.'/lib/ispan.class.php';
|
|
require_once NOALYSS_INCLUDE.'/lib/ihidden.class.php';
|
|
require_once NOALYSS_INCLUDE.'/lib/idate.class.php';
|
|
require_once NOALYSS_INCLUDE.'/lib/itext.class.php';
|
|
require_once NOALYSS_INCLUDE.'/lib/icheckbox.class.php';
|
|
require_once NOALYSS_INCLUDE.'/lib/iperiod.class.php';
|
|
require_once NOALYSS_INCLUDE.'/class/fiche.class.php';
|
|
require_once NOALYSS_INCLUDE.'/class/user.class.php';
|
|
require_once NOALYSS_INCLUDE.'/class/dossier.class.php';
|
|
require_once NOALYSS_INCLUDE.'/class/anc_operation.class.php';
|
|
require_once NOALYSS_INCLUDE.'/class/acc_operation.class.php';
|
|
require_once NOALYSS_INCLUDE.'/class/acc_account_ledger.class.php';
|
|
require_once NOALYSS_INCLUDE.'/class/pre_op_advanced.class.php';
|
|
require_once NOALYSS_INCLUDE.'/class/acc_reconciliation.class.php';
|
|
require_once NOALYSS_INCLUDE.'/class/periode.class.php';
|
|
require_once NOALYSS_INCLUDE.'/class/gestion_purchase.class.php';
|
|
require_once NOALYSS_INCLUDE.'/class/gestion_sold.class.php';
|
|
require_once NOALYSS_INCLUDE.'/class/acc_account.class.php';
|
|
require_once NOALYSS_INCLUDE.'/lib/ac_common.php';
|
|
require_once NOALYSS_INCLUDE.'/lib/inum.class.php';
|
|
require_once NOALYSS_INCLUDE.'/class/lettering.class.php';
|
|
require_once NOALYSS_INCLUDE.'/lib/sort_table.class.php';
|
|
require_once NOALYSS_INCLUDE.'/database/jrn_def_sql.class.php';
|
|
require_once NOALYSS_INCLUDE.'/class/acc_payment.class.php';
|
|
require_once NOALYSS_INCLUDE.'/class/acc_ledger_history.class.php';
|
|
//require_once NOALYSS_INCLUDE.'/class/print_ledger.class.php';
|
|
require_once NOALYSS_INCLUDE.'/lib/http_input.class.php';
|
|
|
|
/** \file
|
|
* @brief Class for jrn, class acc_ledger for manipulating the ledger
|
|
*/
|
|
|
|
/** @brief Class for jrn, class acc_ledger for manipulating the ledger
|
|
*
|
|
*/
|
|
class Acc_Ledger extends jrn_def_sql
|
|
{
|
|
|
|
var $id; /* * < jrn_def.jrn_def_id */
|
|
var $name; /* * < jrn_def.jrn_def_name */
|
|
var $db; /* * < database connextion */
|
|
var $row; /* * < row of the ledger */
|
|
var $type; /* * < type of the ledger ACH ODS FIN
|
|
VEN or GL */
|
|
var $nb; /* * < default number of rows by
|
|
default 10 */
|
|
|
|
/**
|
|
* @param $p_cn database connexion
|
|
* @param $p_id jrn.jrn_def_id
|
|
*/
|
|
function __construct($p_cn, $p_id)
|
|
{
|
|
$this->id=$p_id;
|
|
$this->name=&$this->jrn_def_name;
|
|
$this->jrn_def_id=&$this->id;
|
|
$this->db=$p_cn;
|
|
$this->row=null;
|
|
$this->nb=MAX_ARTICLE;
|
|
}
|
|
|
|
function get_last_pj()
|
|
{
|
|
if (isNumber($this->id)==0)
|
|
{
|
|
throw new Exception(_("Paramètre invalide"));
|
|
return;
|
|
}
|
|
if ($this->db->exist_sequence("s_jrn_pj".$this->id))
|
|
{
|
|
$ret=$this->db->get_array("select last_value,is_called from s_jrn_pj".$this->id);
|
|
$last=$ret[0]['last_value'];
|
|
/**
|
|
* \note With PSQL sequence , the last_value column is 1 when before AND after the first call, to make the difference between them
|
|
* I have to check whether the sequence has been already called or not */
|
|
if ($ret[0]['is_called']=='f')
|
|
$last--;
|
|
return $last;
|
|
}
|
|
else
|
|
$this->db->create_sequence("s_jrn_pj".$this->id);
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* @brief Return the type of a ledger (ACH,VEN,ODS or FIN) or GL
|
|
*
|
|
*/
|
|
function get_type()
|
|
{
|
|
if ($this->id==0)
|
|
{
|
|
$this->name=_(" Tous les journaux");
|
|
$this->type="GL";
|
|
return "GL";
|
|
}
|
|
|
|
$Res=$this->db->exec_sql("select jrn_def_type from ".
|
|
" jrn_def where jrn_def_id=".
|
|
$this->id);
|
|
$Max=Database::num_row($Res);
|
|
if ($Max==0)
|
|
return null;
|
|
$ret=Database::fetch_array($Res, 0);
|
|
$this->type=$ret['jrn_def_type'];
|
|
return $ret['jrn_def_type'];
|
|
}
|
|
|
|
/**
|
|
* let you delete a operation
|
|
* @note by cascade it will delete also in
|
|
* - jrnx
|
|
* - stock
|
|
* - quant_purchase
|
|
* - quant_fin
|
|
* - quant_sold
|
|
* - operation_analytique
|
|
* - letter
|
|
* - reconciliation
|
|
* @bug the attached document is not deleted
|
|
* @bug Normally it should be named delete_operation, cause the id is the ledger_id
|
|
* (jrn_def_id) and not the operation id
|
|
*/
|
|
function delete()
|
|
{
|
|
if ($this->id==0)
|
|
return;
|
|
$grpt_id=$this->db->get_value('select jr_grpt_id from jrn where jr_id=$1',
|
|
array($this->jr_id));
|
|
if ($this->db->count()==0)
|
|
return;
|
|
$this->db->exec_sql('delete from jrnx where j_grpt=$1', array($grpt_id));
|
|
$this->db->exec_sql('delete from jrn where jr_id=$1',
|
|
array($this->jr_id));
|
|
}
|
|
|
|
/**
|
|
* Display warning contained in an array
|
|
* @return string with error message
|
|
*/
|
|
function display_warning($pa_msg, $p_warning)
|
|
{
|
|
$str='<p class="notice"> '.$p_warning;
|
|
$str.="<ol class=\"notice\">";
|
|
for ($i=0; $i<count($pa_msg); $i++)
|
|
{
|
|
$str.="<li>".$pa_msg[$i]."</li>";
|
|
}
|
|
$str.='</ol>';
|
|
$str.='</p>';
|
|
return $str;
|
|
}
|
|
|
|
/**
|
|
* reverse the operation by creating the opposite one,
|
|
* the result is to avoid it
|
|
* it must be done in
|
|
* - jrn
|
|
* - jrnx
|
|
* - quant_fin
|
|
* - quant_sold
|
|
* - quant_purchase
|
|
* - stock
|
|
* - ANC
|
|
* Add or update a note into jrn_note
|
|
* @param $p_date is the date of the reversed op
|
|
* @exception if date is invalid or other prob
|
|
* @note automatically create a reconciliation between operation
|
|
* You must set the ledger_id $this->jrn_def_id
|
|
* This function should be in operation or call an acc_operation object
|
|
*
|
|
*/
|
|
function reverse($p_date,$p_label)
|
|
{
|
|
global $g_user;
|
|
try
|
|
{
|
|
$this->db->start();
|
|
if (!isset($this->jr_id)||$this->jr_id=='')
|
|
throw new Exception(_("this->jr_id is not set ou opération inconnue"));
|
|
|
|
/* check if the date is valid */
|
|
if (isDate($p_date)==null)
|
|
throw new Exception(_('Date invalide').$p_date);
|
|
|
|
// if the operation is in a closed or centralized period
|
|
// the operation is voided thanks the opposite operation
|
|
$grp_new=$this->db->get_next_seq('s_grpt');
|
|
$seq=$this->db->get_next_seq("s_jrn");
|
|
$p_internal=$this->compute_internal_code($seq);
|
|
$this->jr_grpt_id=$this->db->get_value('select jr_grpt_id from jrn where jr_id=$1',
|
|
array($this->jr_id));
|
|
if ($this->db->count()==0)
|
|
throw new Exception(_("Cette opération n'existe pas"));
|
|
$this->jr_internal=$this->db->get_value('select jr_internal from jrn where jr_id=$1',
|
|
array($this->jr_id));
|
|
if ($this->db->count()==0||trim($this->jr_internal)=='')
|
|
throw new Exception(_("Cette opération n'existe pas"));
|
|
|
|
/* find the periode thanks the date */
|
|
$per=new Periode($this->db);
|
|
$per->jrn_def_id=$this->id;
|
|
$per->find_periode($p_date);
|
|
|
|
if ($per->is_open()==0)
|
|
throw new Exception(_('PERIODE FERMEE'));
|
|
|
|
|
|
|
|
|
|
|
|
// Mark the operation invalid into the ledger
|
|
// to avoid to nullify twice the same op., add or update a note into jrn_note
|
|
if ($this->db->get_value("select count(*) from jrn_note where jr_id=$1",[$this->jr_id])>0){
|
|
$sql="update jrn_note set n_text=$2||n_text where jr_id=$1";
|
|
$Res=$this->db->exec_sql($sql, array($this->jr_id,$p_label));
|
|
}else {
|
|
$sql="insert into jrn_note(n_text,jr_id) values ($1,$2)";
|
|
$Res=$this->db->exec_sql($sql, array($p_label,$this->jr_id));
|
|
}
|
|
|
|
// Check return code
|
|
if ($Res==false)
|
|
throw new Exception(__FILE__.__LINE__."sql a echoue [ $sql ]");
|
|
|
|
//////////////////////////////////////////////////
|
|
// Reverse in jrnx* tables
|
|
//////////////////////////////////////////////////
|
|
$a_jid=$this->db->get_array("select j_id,j_debit from jrnx where j_grpt=$1",
|
|
array($this->jr_grpt_id));
|
|
for ($l=0; $l<count($a_jid); $l++)
|
|
{
|
|
$row=$a_jid[$l]['j_id'];
|
|
// Make also the change into jrnx
|
|
$sql="insert into jrnx (
|
|
j_date,j_montant,j_poste,j_grpt,
|
|
j_jrn_def,j_debit,j_text,j_internal,j_tech_user,j_tech_per,j_qcode
|
|
) select to_date($1,'DD.MM.YYYY'),j_montant,j_poste,$2,
|
|
j_jrn_def,not (j_debit),j_text,$3,$4,$5,
|
|
j_qcode
|
|
from
|
|
jrnx
|
|
where j_id=$6 returning j_id";
|
|
$Res=$this->db->exec_sql($sql,
|
|
array($p_date, $grp_new, $p_internal, $g_user->id, $per->p_id,
|
|
$row));
|
|
// Check return code
|
|
if ($Res==false)
|
|
throw (new Exception(__FILE__.__LINE__."SQL ERROR [ $sql ]"));
|
|
$aj_id=$this->db->fetch(0);
|
|
$j_id=$aj_id['j_id'];
|
|
|
|
/* automatic lettering */
|
|
$let=new Lettering($this->db);
|
|
$let->insert_couple($j_id, $row);
|
|
|
|
// reverse in QUANT_SOLD
|
|
$Res=$this->db->exec_sql("INSERT INTO quant_sold(
|
|
qs_internal, qs_fiche, qs_quantite, qs_price, qs_vat,
|
|
qs_vat_code, qs_client, qs_valid, j_id,qs_vat_sided,qs_unit)
|
|
SELECT $1, qs_fiche, qs_quantite*(-1), qs_price*(-1), qs_vat*(-1),
|
|
qs_vat_code, qs_client, qs_valid, $2,qs_vat_sided*(-1),qs_unit*(-1)
|
|
FROM quant_sold where j_id=$3",
|
|
array($p_internal, $j_id, $row));
|
|
|
|
if ($Res==false)
|
|
throw new Exception(__FILE__.__LINE__."sql a echoue [ $sql ]");
|
|
$Res=$this->db->exec_sql("INSERT INTO quant_purchase(
|
|
qp_internal, j_id, qp_fiche, qp_quantite, qp_price, qp_vat,
|
|
qp_vat_code, qp_nd_amount, qp_nd_tva, qp_nd_tva_recup, qp_supplier,
|
|
qp_valid, qp_dep_priv,qp_vat_sided,qp_unit)
|
|
SELECT $1, $2, qp_fiche, qp_quantite*(-1), qp_price*(-1), qp_vat*(-1),
|
|
qp_vat_code, qp_nd_amount*(-1), qp_nd_tva*(-1), qp_nd_tva_recup*(-1), qp_supplier,
|
|
qp_valid, qp_dep_priv*(-1),qp_vat_sided*(-1),qp_unit*(-1)
|
|
FROM quant_purchase where j_id=$3",
|
|
array($p_internal, $j_id, $row));
|
|
|
|
if ($Res==false)
|
|
throw new Exception(__FILE__.__LINE__."SQL ERROR [ $sql ]");
|
|
}
|
|
$sql="insert into jrn (
|
|
jr_id,
|
|
jr_def_id,
|
|
jr_montant,
|
|
jr_comment,
|
|
jr_date,
|
|
jr_grpt_id,
|
|
jr_internal
|
|
,jr_tech_per, jr_valid,jr_optype
|
|
)
|
|
select $1,jr_def_id,jr_montant,$7,
|
|
to_date($2,'DD.MM.YYYY'),$3,$4,
|
|
$5, true,'EXT'
|
|
from
|
|
jrn
|
|
where jr_id=$6";
|
|
$Res=$this->db->exec_sql($sql,
|
|
array($seq, $p_date, $grp_new, $p_internal, $per->p_id, $this->jr_id,$p_label));
|
|
// Check return code
|
|
if ($Res==false)
|
|
throw (new Exception(__FILE__.__LINE__."SQL ERROR [ $sql ]"));
|
|
// reverse in QUANT_FIN table
|
|
$Res=$this->db->exec_sql(" INSERT INTO quant_fin(
|
|
qf_bank, qf_other, qf_amount,jr_id)
|
|
SELECT qf_bank, qf_other, qf_amount*(-1),$1
|
|
FROM quant_fin where jr_id=$2",
|
|
array($seq, $this->jr_id));
|
|
if ($Res==false)
|
|
throw (new Exception(__FILE__.__LINE__."SQL ERROR[ $sql ]"));
|
|
|
|
// Add a "concerned operation to bound these op.together
|
|
//
|
|
$rec=new Acc_Reconciliation($this->db);
|
|
$rec->set_jr_id($seq);
|
|
$rec->insert($this->jr_id);
|
|
|
|
// Check return code
|
|
if ($Res==false)
|
|
{
|
|
throw (new Exception(__FILE__.__LINE__."SQL ERROR [ $sql ]"));
|
|
}
|
|
|
|
|
|
|
|
// the table stock must updated
|
|
// also in the stock table
|
|
$sql="delete from stock_goods where sg_id = any ( select sg_id
|
|
from stock_goods natural join jrnx where j_grpt=".$this->jr_grpt_id.")";
|
|
$Res=$this->db->exec_sql($sql);
|
|
if ($Res==false)
|
|
throw (new Exception(__FILE__.__LINE__."SQL ERROR [ $sql ]"));
|
|
$this->db->commit();
|
|
}
|
|
catch (Exception $e)
|
|
{
|
|
record_log($e->getTraceAsString());
|
|
$this->db->rollback();
|
|
throw $e;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @brief Return the name of a ledger
|
|
*
|
|
*/
|
|
function get_name()
|
|
{
|
|
if ($this->id==0)
|
|
{
|
|
$this->name=_("Grand Livre");
|
|
return $this->name;
|
|
}
|
|
|
|
$Res=$this->db->exec_sql("select jrn_def_name from ".
|
|
" jrn_def where jrn_def_id=$1", array($this->id));
|
|
$Max=Database::num_row($Res);
|
|
if ($Max==0)
|
|
return null;
|
|
$ret=Database::fetch_array($Res, 0);
|
|
$this->name=$ret['jrn_def_name'];
|
|
return $ret['jrn_def_name'];
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
* @brief guess what the next pj should be
|
|
*/
|
|
function guess_pj()
|
|
{
|
|
$prop=$this->get_propertie();
|
|
$pj_pref=$prop["jrn_def_pj_pref"];
|
|
$pj_seq=$this->get_last_pj()+1;
|
|
return $pj_pref.$pj_seq;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// retrieve data from jrnx
|
|
/**
|
|
* @brief Get the properties of a journal
|
|
*
|
|
* \return an array containing properties
|
|
*
|
|
*/
|
|
function get_propertie()
|
|
{
|
|
if ($this->id==0)
|
|
return;
|
|
|
|
$Res=$this->db->exec_sql("select jrn_Def_id,jrn_def_name,jrn_def_class_deb,jrn_def_class_cred,jrn_def_type,
|
|
jrn_deb_max_line,jrn_cred_max_line,jrn_def_ech,jrn_def_ech_lib,jrn_def_code,
|
|
jrn_def_fiche_deb,jrn_def_fiche_cred,jrn_def_pj_pref
|
|
from jrn_Def
|
|
where jrn_def_id=$1", array($this->id));
|
|
$Count=Database::num_row($Res);
|
|
if ($Count==0)
|
|
{
|
|
echo '<DIV="redcontent"><H2 class="error">'._('Parametres journaux non trouves').'</H2> </DIV>';
|
|
return null;
|
|
}
|
|
return Database::fetch_array($Res, 0);
|
|
}
|
|
|
|
/**
|
|
* @brief Get the number of lines of a journal
|
|
* @param$p_cred deb or cred
|
|
*
|
|
* \return an integer
|
|
*/
|
|
function GetDefLine()
|
|
{
|
|
$sql_cred='jrn_deb_max_line';
|
|
$sql="select jrn_deb_max_line as value from jrn_def where jrn_def_id=$1";
|
|
$r=$this->db->exec_sql($sql, array($this->id));
|
|
$Res=Database::fetch_all($r);
|
|
if (sizeof($Res)==0)
|
|
return 1;
|
|
return $Res[0]['value'];
|
|
}
|
|
|
|
/**
|
|
* @brief get the saldo of a ledger for a specific period
|
|
* @param$p_from start period
|
|
* @param$p_to end period
|
|
*/
|
|
function get_solde($p_from, $p_to)
|
|
{
|
|
$ledger="";
|
|
if ($this->id!=0)
|
|
{
|
|
$ledger=" and j_jrn_def = ".$this->id;
|
|
}
|
|
|
|
$periode=sql_filter_per($this->db, $p_from, $p_to, 'p_id', 'j_tech_per');
|
|
$sql='select j_montant as montant,j_debit as deb from jrnx where '
|
|
.$periode.$ledger;
|
|
|
|
$ret=$this->db->exec_sql($sql);
|
|
$array=Database::fetch_all($ret);
|
|
$deb=0.0;
|
|
$cred=0.0;
|
|
foreach ($array as $line)
|
|
{
|
|
|
|
if ($line['deb']=='t')
|
|
$deb+=$line['montant'];
|
|
else
|
|
$cred+=$line['montant'];
|
|
}
|
|
$response=array($deb, $cred);
|
|
return $response;
|
|
}
|
|
|
|
/**
|
|
* @brief Show a select list of the ledgers you can access in
|
|
* writing, reading or simply accessing.
|
|
* @param $p_type = ALL or the type of the ledger (ACH,VEN,FIN,ODS)
|
|
* @param $p_access =3 for READ and WRITE, 2 for write and 1 for readonly
|
|
* @param Boolean TRUE all ledger are selected, or FALSE only enable
|
|
* \return object HtmlInput select
|
|
*
|
|
*/
|
|
function select_ledger($p_type="ALL", $p_access=3,$enable=TRUE)
|
|
{
|
|
global $g_user;
|
|
$array=$g_user->get_ledger($p_type, $p_access,$enable);
|
|
|
|
if ($array==null)
|
|
return null;
|
|
$idx=0;
|
|
$ret=array();
|
|
|
|
foreach ($array as $value)
|
|
{
|
|
$ret[$idx]['value']=$value['jrn_def_id'];
|
|
$ret[$idx]['label']=h($value['jrn_def_name']);
|
|
$idx++;
|
|
}
|
|
|
|
$select=new ISelect();
|
|
$select->name='p_jrn';
|
|
$select->value=$ret;
|
|
$select->selected=$this->id;
|
|
return $select;
|
|
}
|
|
|
|
/**
|
|
* @brief retrieve the jrn_def_fiche and return them into a array
|
|
* index deb, cred
|
|
* \param
|
|
* \param
|
|
* \param
|
|
*
|
|
*
|
|
* \return return an array ('deb'=> ,'cred'=>)
|
|
*/
|
|
function get_fiche_def()
|
|
{
|
|
$sql="select jrn_def_fiche_deb as deb,jrn_def_fiche_cred as cred ".
|
|
" from jrn_def where ".
|
|
" jrn_def_id = $1 ";
|
|
|
|
$r=$this->db->exec_sql($sql, array($this->id));
|
|
|
|
$res=Database::fetch_all($r);
|
|
if (empty($res))
|
|
return null;
|
|
|
|
return $res[0];
|
|
}
|
|
|
|
/**
|
|
* @brief retrieve the jrn_def_class_deb and return it
|
|
*
|
|
*
|
|
* \return return an string
|
|
*/
|
|
function get_class_def()
|
|
{
|
|
$sql="select jrn_def_class_deb ".
|
|
" from jrn_def where ".
|
|
" jrn_def_id = $1";
|
|
|
|
$r=$this->db->exec_sql($sql, array($this->id));
|
|
|
|
$res=Database::fetch_all($r);
|
|
|
|
if (empty($res))
|
|
return null;
|
|
|
|
return $res[0];
|
|
}
|
|
|
|
/**
|
|
* @brief show the result of the array to confirm
|
|
* before inserting
|
|
* @param$p_array array from the form
|
|
* @return HTML string
|
|
*/
|
|
function confirm($p_array, $p_readonly=false)
|
|
{
|
|
global $g_parameter;
|
|
$msg=array();
|
|
if (!$p_readonly)
|
|
$msg=$this->verify($p_array);
|
|
$this->id=$p_array['p_jrn'];
|
|
if (empty($p_array))
|
|
return _("Aucun résultat");
|
|
$anc=null;
|
|
extract($p_array, EXTR_SKIP);
|
|
$lPeriode=new Periode($this->db);
|
|
if ($this->check_periode()==true)
|
|
{
|
|
$lPeriode->p_id=$period;
|
|
}
|
|
else
|
|
{
|
|
$lPeriode->find_periode($e_date);
|
|
}
|
|
$total_deb=0;
|
|
$total_cred=0;
|
|
bcscale(2);
|
|
|
|
$ret="";
|
|
if (!empty($msg))
|
|
{
|
|
$ret.=$this->display_warning($msg,
|
|
_("Attention : il vaut mieux utiliser les fiches que les postes comptables"));
|
|
}
|
|
$ret.="<table >";
|
|
$ret.="<tr><td>"._('Date')." : </td><td>$e_date</td></tr>";
|
|
/* display periode */
|
|
$date_limit=$lPeriode->get_date_limit();
|
|
$ret.='<tr> '.td(_('Période Comptable')).td($date_limit['p_start'].'-'.$date_limit['p_end']).'</tr>';
|
|
$ret.="<tr><td>"._('Libellé')." </td><td>".h($desc)."</td></tr>";
|
|
$ret.="<tr><td>"._('PJ Num')." </td><td>".h($e_pj)."</td></tr>";
|
|
$ret.='</table>';
|
|
$ret.="<table class=\"result\">";
|
|
$ret.="<tr>";
|
|
$ret.="<th>"._('Quick Code ou ');
|
|
$ret.=_("Poste")." </th>";
|
|
$ret.="<th style=\"text-align:left\"> "._("Libellé")." </th>";
|
|
$ret.="<th style=\"text-align:right\">"._("Débit")."</th>";
|
|
$ret.="<th style=\"text-align:right\">"._("Crédit")."</th>";
|
|
/* if we use the AC */
|
|
if ($g_parameter->MY_ANALYTIC!='nu')
|
|
{
|
|
$anc=new Anc_Plan($this->db);
|
|
$a_anc=$anc->get_list();
|
|
$x=count($a_anc);
|
|
/* set the width of the col */
|
|
$ret.='<th colspan="'.$x.'" style="width:auto;text-align:center" >'._('Compt. Analytique').'</th>';
|
|
|
|
/* add hidden variables pa[] to hold the value of pa_id */
|
|
$ret.=Anc_Plan::hidden($a_anc);
|
|
}
|
|
$ret.="</tr>";
|
|
|
|
$ret.=HtmlInput::hidden('e_date', $e_date);
|
|
$ret.=HtmlInput::hidden('desc', $desc);
|
|
$ret.=HtmlInput::hidden('period', $lPeriode->p_id);
|
|
$ret.=HtmlInput::hidden('e_pj', $e_pj);
|
|
$ret.=HtmlInput::hidden('e_pj_suggest', $e_pj_suggest);
|
|
$mt=microtime(true);
|
|
$ret.=HtmlInput::hidden('mt', $mt);
|
|
// For predefined operation
|
|
$ret.=HtmlInput::hidden('e_comm', $desc);
|
|
$ret.=HtmlInput::hidden('jrn_type', $this->get_type());
|
|
$ret.=HtmlInput::hidden('p_jrn', $this->id);
|
|
$ret.=HtmlInput::hidden('nb_item', $nb_item);
|
|
if ($this->with_concerned==true)
|
|
{
|
|
$ret.=HtmlInput::hidden('jrn_concerned', $jrn_concerned);
|
|
}
|
|
$ret.=dossier::hidden();
|
|
$count=0;
|
|
for ($i=0; $i<$nb_item; $i++)
|
|
{
|
|
if ($p_readonly==true)
|
|
{
|
|
if (!isset(${'qc_'.$i}))
|
|
${'qc_'.$i}='';
|
|
if (!isset(${'poste'.$i}))
|
|
${'poste'.$i}='';
|
|
if (!isset(${'amount'.$i}))
|
|
${'amount'.$i}='';
|
|
}
|
|
$class=($i%2==0)?' class="even" ':' class="odd" ';
|
|
$ret.="<tr $class> ";
|
|
if (trim(${'qc_'.$i})!="")
|
|
{
|
|
$oqc=new Fiche($this->db);
|
|
$oqc->get_by_qcode(${'qc_'.$i}, false);
|
|
$strPoste=$oqc->strAttribut(ATTR_DEF_ACCOUNT);
|
|
$ret.="<td>".
|
|
${'qc_'.$i}.' - '.
|
|
$oqc->strAttribut(ATTR_DEF_NAME).HtmlInput::hidden('qc_'.$i,
|
|
${'qc_'.$i}).
|
|
'</td>';
|
|
}
|
|
|
|
if (trim(${'qc_'.$i})==""&&trim(${'poste'.$i})!="")
|
|
{
|
|
$oposte=new Acc_Account_Ledger($this->db, ${'poste'.$i});
|
|
$strPoste=$oposte->id;
|
|
$ret.="<td>".h(${"poste".$i}." - ".
|
|
$oposte->get_name()).HtmlInput::hidden('poste'.$i,
|
|
${'poste'.$i}).
|
|
'</td>';
|
|
}
|
|
|
|
if (trim(${'qc_'.$i})==""&&trim(${'poste'.$i})=="")
|
|
continue;
|
|
$ret.="<td>".h(${"ld".$i}).HtmlInput::hidden('ld'.$i, ${'ld'.$i});
|
|
$ret.=(isset(${"ck$i"}))?HtmlInput::hidden('ck'.$i, ${'ck'.$i}):"";
|
|
$ret.="</td>";
|
|
if (isset(${"ck$i"}))
|
|
{
|
|
$ret.="<td class=\"num\">".nbm(${"amount".$i}).HtmlInput::hidden('amount'.$i,
|
|
${'amount'.$i})."</td>".td("");
|
|
$total_deb=bcadd($total_deb, ${'amount'.$i});
|
|
}
|
|
else
|
|
{
|
|
$ret.=td("")."<td class=\"num\">".nbm(${"amount".$i}).HtmlInput::hidden('amount'.$i,
|
|
${'amount'.$i})."</td>";
|
|
$total_cred=bcadd($total_cred, ${"amount".$i});
|
|
}
|
|
// CA
|
|
if ($g_parameter->MY_ANALYTIC!='nu') // use of AA
|
|
{
|
|
if (preg_match("/^[6,7]+/", $strPoste)==1)
|
|
{
|
|
// show form
|
|
$op=new Anc_Operation($this->db);
|
|
$null=($g_parameter->MY_ANALYTIC=='op')?1:0;
|
|
$p_array['pa_id']=$a_anc;
|
|
/* op is the operation it contains either a sequence or a jrnx.j_id */
|
|
$ret.=HtmlInput::hidden('op[]=', $i);
|
|
|
|
$ret.='<td style="text-align:center">';
|
|
$read=($p_readonly==true)?0:1;
|
|
$ret.=$op->display_form_plan($p_array, $null, $read, $count,
|
|
round(${'amount'.$i}, 2));
|
|
$ret.='</td>';
|
|
$count++;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
$ret.="</tr>";
|
|
}
|
|
$ret.=tr(td('').td(_('Totaux')).td($total_deb, 'class="num"').td($total_cred,
|
|
'class="num"'), 'class="highlight"');
|
|
$ret.="</table>";
|
|
if ($g_parameter->MY_ANALYTIC!='nu'&&$p_readonly==false)
|
|
$ret.='<input type="button" class="button" value="'._('verifie Imputation Analytique').'" onClick="verify_ca(\'\');">';
|
|
return $ret;
|
|
}
|
|
|
|
function get_min_row()
|
|
{
|
|
$row=$this->db->get_value("select jrn_deb_max_line from jrn_def where jrn_def_id=$1",
|
|
array($this->id));
|
|
return $row;
|
|
}
|
|
|
|
/**
|
|
* @brief Show the form to encode your operation
|
|
* @param$p_array if you correct or use a predef operation (default = null)
|
|
* @param$p_readonly 1 for readonly 0 for writable (default 0)
|
|
* @exception if ledger not found
|
|
* \return a string containing the form
|
|
*/
|
|
function input($p_array=null, $p_readonly=0)
|
|
{
|
|
global $g_parameter, $g_user;
|
|
$this->nb=$this->get_min_row();
|
|
if ($p_readonly==1)
|
|
return $this->confirm($p_array);
|
|
|
|
if ($p_array!=null)
|
|
extract($p_array, EXTR_SKIP);
|
|
$add_js="";
|
|
if ($g_parameter->MY_PJ_SUGGEST=='Y')
|
|
{
|
|
$add_js="update_pj();";
|
|
}
|
|
if ($g_parameter->MY_DATE_SUGGEST=='Y')
|
|
{
|
|
$add_js.='get_last_date();';
|
|
}
|
|
$add_js.='update_row("quick_item");';
|
|
$ret="";
|
|
if ($g_user->check_action(FICADD)==1)
|
|
{
|
|
// Button for adding customer
|
|
$add_card=TRUE;
|
|
}
|
|
$wLedger=$this->select_ledger('ODS', 2,FALSE);
|
|
if ($wLedger==null)
|
|
throw new Exception(_('Pas de journal disponible'));
|
|
$wLedger->javascript="onChange='update_name();update_predef(\"ods\",\"t\",\"".$_REQUEST['ac']."\");$add_js'";
|
|
$label=" Journal ".Icon_Action::infobulle(2);
|
|
|
|
$ret.="<table>";
|
|
$ret.=tr(td($label).td($wLedger->input()));
|
|
|
|
//
|
|
// Button for template operation
|
|
//
|
|
ob_start();
|
|
echo '<div id="predef_form">';
|
|
echo HtmlInput::hidden('p_jrn_predef', $this->id);
|
|
$op=new Pre_op_ods($this->db);
|
|
$op->set('ledger', $this->id);
|
|
$op->set('ledger_type', "ODS");
|
|
$op->set('direct', 't');
|
|
$url=http_build_query(array('action'=>'use_opd', 'p_jrn_predef'=>$this->id,
|
|
'ac'=>$_REQUEST['ac'], 'gDossier'=>dossier::id()));
|
|
echo $op->form_get('do.php?'.$url);
|
|
|
|
echo '</div>';
|
|
$str_op_template=ob_get_contents();
|
|
ob_end_clean();
|
|
$ret.="<tr>";
|
|
$ret.="<td>"._("Modèle d'opération")."</td>";
|
|
$ret.="<td>".$str_op_template."</td>";
|
|
$ret.="</tr>";
|
|
// Load the javascript
|
|
//
|
|
//$ret.= '<tr ><td colspan="2" style="width:auto">';
|
|
$wDate=new IDate('e_date');
|
|
$wDate->readonly=$p_readonly;
|
|
$e_date=(isset($e_date)&&trim($e_date)!='')?$e_date:'';
|
|
$wDate->value=$e_date;
|
|
|
|
$ret.=tr(td(_("Date")).td($wDate->input()));
|
|
/* insert periode if needed */
|
|
// Periode
|
|
//--
|
|
if ($this->check_periode()==true)
|
|
{
|
|
$l_user_per=$g_user->get_periode();
|
|
$def=(isset($periode))?$periode:$l_user_per;
|
|
|
|
$period=new IPeriod("period");
|
|
$period->user=$g_user;
|
|
$period->cn=$this->db;
|
|
$period->value=$def;
|
|
$period->type=OPEN;
|
|
try
|
|
{
|
|
$l_form_per=$period->input();
|
|
}
|
|
catch (Exception $e)
|
|
{
|
|
record_log($e->getTraceAsString());
|
|
if ($e->getCode()==1)
|
|
{
|
|
echo _("Aucune période ouverte");
|
|
exit();
|
|
}
|
|
}
|
|
$label=Icon_Action::infobulle(3);
|
|
$f_periode=td(_("Période comptable")." $label ").td($l_form_per);
|
|
$ret.=tr($f_periode);
|
|
}
|
|
$wPJ=new IText('e_pj');
|
|
$wPJ->readonly=false;
|
|
$wPJ->size=10;
|
|
|
|
/* suggest PJ ? */
|
|
$default_pj='';
|
|
if ($g_parameter->MY_PJ_SUGGEST=='Y')
|
|
{
|
|
$default_pj=$this->guess_pj();
|
|
}
|
|
$wPJ->value=(isset($e_pj))?$e_pj:$default_pj;
|
|
$ret.='</tr>';
|
|
$ret.='<tr >';
|
|
$ret.='<td colspan="2" style="width:auto"> '._('Pièce').' : '.$wPJ->input();
|
|
$ret.=HtmlInput::hidden('e_pj_suggest', $default_pj);
|
|
$ret.='</tr>';
|
|
$ret.='</td>';
|
|
|
|
$ret.='<tr>';
|
|
$ret.='<td colspan="2" style="width:auto">';
|
|
$ret.=_('Libellé');
|
|
$wDescription=new IText('desc');
|
|
$wDescription->readonly=$p_readonly;
|
|
$wDescription->size="50";
|
|
$wDescription->value=(isset($desc))?$desc:'';
|
|
|
|
$ret.=$wDescription->input();
|
|
$ret.='</td>';
|
|
$ret.='</tr>';
|
|
|
|
$ret.='</table>';
|
|
$nb_row=(isset($nb_item) )?$nb_item:$this->nb;
|
|
|
|
$ret.=HtmlInput::hidden('nb_item', $nb_row);
|
|
$ret.=dossier::hidden();
|
|
$ret.=HtmlInput::hidden('jrn_type', $this->get_type());
|
|
$info=Icon_Action::infobulle(0);
|
|
$info_poste=Icon_Action::infobulle(9);
|
|
$ret.='<table id="quick_item" style="position:float;width:100%">';
|
|
$ret.='<tr>'.
|
|
'<th style="text-align:left">Quickcode'.$info.'</th>'.
|
|
'<th style="text-align:left">'._('Poste').$info_poste.'</th>'.
|
|
'<th class="visible_gt800 visible_gt1155" style="text-align:left">'._('Libellé').'</th>'.
|
|
'<th style="text-align:left">'._('Montant').'</th>'.
|
|
'<th style="text-align:left">'._('Débit').'</th>'.
|
|
'</tr>';
|
|
|
|
|
|
for ($i=0; $i<$nb_row; $i++)
|
|
{
|
|
// Quick Code
|
|
$quick_code=new ICard('qc_'.$i);
|
|
$quick_code->set_dblclick("fill_ipopcard(this);");
|
|
$quick_code->set_attribute('ipopup', 'ipopcard');
|
|
|
|
// name of the field to update with the name of the card
|
|
$quick_code->set_attribute('label', "ld".$i);
|
|
|
|
// name of the field to update with the name of the card
|
|
$quick_code->set_attribute('typecard', 'filter');
|
|
|
|
// Add the callback function to filter the card on the jrn
|
|
$quick_code->set_callback('filter_card');
|
|
$quick_code->set_function('fill_data');
|
|
$quick_code->javascript=sprintf(' onchange="fill_data_onchange(\'%s\');" ',
|
|
$quick_code->name);
|
|
|
|
$quick_code->value=(isset(${'qc_'.$i}))?${'qc_'.$i}:"";
|
|
$quick_code->readonly=$p_readonly;
|
|
|
|
$label='';
|
|
if ($quick_code->value!='')
|
|
{
|
|
$Fiche=new Fiche($this->db);
|
|
$Fiche->get_by_qcode($quick_code->value);
|
|
$label=$Fiche->strAttribut(ATTR_DEF_NAME);
|
|
}
|
|
|
|
|
|
// Account
|
|
$poste=new IPoste();
|
|
$poste->name='poste'.$i;
|
|
$poste->set_attribute('jrn', $this->id);
|
|
$poste->set_attribute('ipopup', 'ipop_account');
|
|
$poste->set_attribute('label', 'ld'.$i);
|
|
$poste->set_attribute('account', 'poste'.$i);
|
|
$poste->set_attribute('dossier', Dossier::id());
|
|
|
|
$poste->value=(isset(${'poste'.$i}))?${"poste".$i}:''
|
|
;
|
|
$poste->dbl_click_history();
|
|
|
|
$poste->readonly=$p_readonly;
|
|
|
|
if ($poste->value!='')
|
|
{
|
|
$Poste=new Acc_Account($this->db,$poste->value);
|
|
$label=$Poste->get_lib();
|
|
}
|
|
|
|
// Description of the line
|
|
$line_desc=new IText();
|
|
$line_desc->name='ld'.$i;
|
|
$line_desc->size=30;
|
|
$line_desc->value=(isset(${"ld".$i}))?${"ld".$i}:
|
|
$label;
|
|
|
|
// Amount
|
|
$amount=new INum();
|
|
$amount->size=10;
|
|
$amount->name='amount'.$i;
|
|
$amount->value=(isset(${'amount'.$i}))?${"amount".$i}:''
|
|
;
|
|
$amount->readonly=$p_readonly;
|
|
$amount->javascript=' onChange="format_number(this);checkTotalDirect()"';
|
|
// D/C
|
|
$deb=new ICheckBox();
|
|
$deb->name='ck'.$i;
|
|
$deb->selected=(isset(${'ck'.$i}))?true:false;
|
|
$deb->readonly=$p_readonly;
|
|
$deb->javascript=' onChange="checkTotalDirect()"';
|
|
$str_add_button=($add_card==TRUE)?$this->add_card("-1",
|
|
$quick_code->id):"";
|
|
$ret.='<tr>';
|
|
$ret.='<td>'.$quick_code->input().$quick_code->search().$str_add_button.'</td>';
|
|
$ret.='<td>'.$poste->input().
|
|
'<script> document.getElementById(\'poste'.$i.'\').onblur=function(){ if (trim(this.value) !=\'\') {document.getElementById(\'qc_'.$i.'\').value="";}}</script>'.
|
|
'</td>';
|
|
$ret.='<td class="visible_gt800 visible_gt1155">'.$line_desc->input().'</td>';
|
|
$ret.='<td>'.$amount->input().'</td>';
|
|
$ret.='<td>'.$deb->input().'</td>';
|
|
$ret.='</tr>';
|
|
// If readonly == 1 then show CA
|
|
}
|
|
$ret.='</table>';
|
|
if (isset($this->with_concerned)&&$this->with_concerned==true)
|
|
{
|
|
$oRapt=new Acc_Reconciliation($this->db);
|
|
$w=$oRapt->widget();
|
|
$w->name='jrn_concerned';
|
|
$w->value=(isset($jrn_concerned))?$jrn_concerned:"";
|
|
$ret.=sprintf(_("Réconciliation/rapprochements : %s"), $w->input());
|
|
}
|
|
$ret.=create_script("$('".$wDate->id."').focus()");
|
|
return $ret;
|
|
}
|
|
|
|
/**
|
|
* @brief
|
|
* check if the current ledger is closed
|
|
* \return 1 for yes, otherwise 0
|
|
* \see Periode::is_closed
|
|
*/
|
|
function is_closed($p_periode)
|
|
{
|
|
$per=new Periode($this->db);
|
|
$per->set_ledger($this->id);
|
|
$per->set_periode($p_periode);
|
|
$ret=$per->is_closed();
|
|
return $ret;
|
|
}
|
|
|
|
/**
|
|
* @brief verify that the operation can be saved
|
|
* @param$p_array array of data same layout that the $_POST from show_form
|
|
*
|
|
*
|
|
* \throw the getcode value is 1 incorrect balance, 2 date
|
|
* invalid, 3 invalid amount, 4 the card is not in the range of
|
|
* permitted card, 5 not in the user's period, 6 closed period
|
|
*
|
|
*/
|
|
function verify($p_array)
|
|
{
|
|
if (is_array($p_array)==false||empty($p_array))
|
|
throw new Exception("Array empty");
|
|
/*
|
|
* Check needed value
|
|
*/
|
|
check_parameter($p_array, 'p_jrn,e_date');
|
|
|
|
extract($p_array, EXTR_SKIP);
|
|
global $g_user;
|
|
$tot_cred=0;
|
|
$tot_deb=0;
|
|
$msg=array();
|
|
|
|
/* check if we can write into this ledger */
|
|
if ($g_user->check_jrn($p_jrn)!='W')
|
|
throw new Exception(_('Accès interdit'), 20);
|
|
|
|
/* check for a double reload */
|
|
if (isset($mt)&&$this->db->count_sql('select jr_mt from jrn where jr_mt=$1',
|
|
array($mt))!=0)
|
|
throw new Exception('Double Encodage', 5);
|
|
|
|
// Check the periode and the date
|
|
if (isDate($e_date)==null)
|
|
{
|
|
throw new Exception('Date invalide', 2);
|
|
}
|
|
$periode=new Periode($this->db);
|
|
/* find the periode if we have enabled the check_periode
|
|
* or if period is not set
|
|
*/
|
|
if ($this->check_periode()==false||!isset($p_array['period']))
|
|
{
|
|
$periode->find_periode($e_date);
|
|
}
|
|
else
|
|
{
|
|
$periode->p_id=$p_array['period'];
|
|
list ($min, $max)=$periode->get_date_limit();
|
|
if (cmpDate($e_date, $min)<0||
|
|
cmpDate($e_date, $max)>0)
|
|
throw new Exception(_('Date et periode ne correspondent pas'), 6);
|
|
}
|
|
|
|
|
|
|
|
// Periode ferme
|
|
if ($this->is_closed($periode->p_id)==1)
|
|
{
|
|
throw new Exception('Periode fermee', 6);
|
|
}
|
|
/* check if we are using the strict mode */
|
|
if ($this->check_strict()==true)
|
|
{
|
|
/* if we use the strict mode, we get the date of the last
|
|
operation */
|
|
$last_date=$this->get_last_date();
|
|
if ($last_date!=null&&cmpDate($e_date, $last_date)<0)
|
|
throw new Exception(_('Vous utilisez le mode strict la dernière operation est la date du ')
|
|
.$last_date.' '._('vous ne pouvez pas encoder à une date antérieure'),
|
|
15);
|
|
}
|
|
|
|
for ($i=0; $i<$nb_item; $i++)
|
|
{
|
|
$err=0;
|
|
|
|
// Check the balance
|
|
if (!isset(${'amount'.$i}))
|
|
continue;
|
|
|
|
$amount=round(${'amount'.$i}, 2);
|
|
$tot_deb+=(isset(${'ck'.$i}))?$amount:0;
|
|
$tot_cred+=(!isset(${'ck'.$i}))?$amount:0;
|
|
|
|
// Check if the card is permitted
|
|
if (isset(${'qc_'.$i})&&trim(${'qc_'.$i})!="")
|
|
{
|
|
$f=new Fiche($this->db);
|
|
$f->quick_code=${'qc_'.$i};
|
|
if ($f->belong_ledger($p_jrn)<0)
|
|
throw new Exception("La fiche quick_code = ".
|
|
$f->quick_code." n'est pas dans ce journal", 4);
|
|
if (strlen(trim(${'qc_'.$i}))!=0&&isNumber(${'amount'.$i})==0)
|
|
throw new Exception('Montant invalide', 3);
|
|
|
|
$strPoste=$f->strAttribut(ATTR_DEF_ACCOUNT);
|
|
if ($strPoste=='')
|
|
throw new Exception(sprintf(_("La fiche %s n'a pas de poste comptable"),
|
|
${"qc_".$i}));
|
|
|
|
$p=new Acc_Account_Ledger($this->db, $strPoste);
|
|
if ($p->do_exist()==0)
|
|
throw new Exception(_('Poste Inexistant pour la fiche ['.${'qc_'.$i}.']'),
|
|
4);
|
|
}
|
|
|
|
// Check if the account is permitted
|
|
if (isset(${'poste'.$i})&&strlen(trim(${'poste'.$i}))!=0)
|
|
{
|
|
$p=new Acc_Account_Ledger($this->db, ${'poste'.$i});
|
|
if ($p->belong_ledger($p_jrn)<0)
|
|
throw new Exception(_("Le poste")." ".$p->id." "._("n'est pas dans ce journal"),
|
|
5);
|
|
if (strlen(trim(${'poste'.$i}))!=0&&isNumber(${'amount'.$i})==0)
|
|
throw new Exception(_('Poste invalide ['.${'poste'.$i}.']'),
|
|
3);
|
|
if ($p->do_exist()==0)
|
|
throw new Exception(_('Poste Inexistant ['.${'poste'.$i}.']'),
|
|
4);
|
|
$card_id=$p->find_card();
|
|
if (!empty($card_id))
|
|
{
|
|
$str_msg=" Le poste ".$p->id." appartient à ".count($card_id)." fiche(s) dont :";
|
|
$max=(count($card_id)>MAX_COMPTE_CARD)?MAX_COMPTE_CARD:count($card_id);
|
|
for ($x=0; $x<$max; $x++)
|
|
{
|
|
$card=new Fiche($this->db, $card_id[$x]['f_id']);
|
|
$str_msg.=HtmlInput::card_detail($card->strAttribut(ATTR_DEF_QUICKCODE),
|
|
$card->strAttribut(ATTR_DEF_NAME),
|
|
'style="color:red;display:inline;text-decoration:underline"');
|
|
$str_msg.=" ";
|
|
}
|
|
$msg[]=$str_msg;
|
|
}
|
|
$account=new Acc_Account($this->db,${"poste".$i});
|
|
if ( $account->get_parameter("pcm_direct_use") == "N") {
|
|
throw new Exception(_("Utilisation directe interdite du poste comptable ${"poste".$i}"));
|
|
}
|
|
}
|
|
}
|
|
$tot_deb=round($tot_deb, 4);
|
|
$tot_cred=round($tot_cred, 4);
|
|
if ($tot_deb!=$tot_cred)
|
|
{
|
|
throw new Exception(_("Balance incorrecte ")." debit = $tot_deb credit=$tot_cred ",
|
|
1);
|
|
}
|
|
|
|
return $msg;
|
|
}
|
|
|
|
/**
|
|
* @brief compute the internal code of the saved operation and set the $this->jr_internal to
|
|
* the computed value
|
|
*
|
|
* @param$p_grpt id in jr_grpt_
|
|
*
|
|
* \return string internal_code
|
|
* -
|
|
*
|
|
*/
|
|
function compute_internal_code($p_grpt)
|
|
{
|
|
if ($this->id==0)
|
|
return;
|
|
$num=$this->db->get_next_seq('s_internal');
|
|
$atype=$this->get_propertie();
|
|
$type=substr($atype['jrn_def_code'], 0, 1);
|
|
$internal_code=sprintf("%s%06X", $type, $num);
|
|
$this->jr_internal=$internal_code;
|
|
return $internal_code;
|
|
}
|
|
|
|
/**
|
|
* @brief save the operation into the jrnx,jrn, ,
|
|
* CA and pre_def
|
|
* @param$p_array
|
|
*
|
|
* \return array with [0] = false if failed otherwise true, [1] error
|
|
* code
|
|
*/
|
|
function save($p_array=null)
|
|
{
|
|
if ($p_array==null)
|
|
throw new Exception('save cannot use a empty array');
|
|
global $g_parameter;
|
|
extract($p_array, EXTR_SKIP);
|
|
try
|
|
{
|
|
$msg=$this->verify($p_array);
|
|
if (!empty($msg))
|
|
{
|
|
echo $this->display_warning($msg,
|
|
_("Attention : il vaut mieux utiliser les fiches que les postes comptables "));
|
|
}
|
|
$this->db->start();
|
|
|
|
$seq=$this->db->get_next_seq('s_grpt');
|
|
$internal=$this->compute_internal_code($seq);
|
|
|
|
$group=$this->db->get_next_seq("s_oa_group");
|
|
$tot_amount=0;
|
|
$tot_deb=0;
|
|
$tot_cred=0;
|
|
$oPeriode=new Periode($this->db);
|
|
$check_periode=$this->check_periode();
|
|
if ($check_periode==false||!isset($p_array['period']))
|
|
{
|
|
$oPeriode->find_periode($e_date);
|
|
}
|
|
else
|
|
{
|
|
$oPeriode->id=$period;
|
|
}
|
|
|
|
$count=0;
|
|
for ($i=0; $i<$nb_item; $i++)
|
|
{
|
|
if (!isset(${'qc_'.$i})&&!isset(${'poste'.$i}))
|
|
continue;
|
|
$acc_op=new Acc_Operation($this->db);
|
|
$quick_code="";
|
|
// First we save the jrnx
|
|
if (isset(${'qc_'.$i}))
|
|
{
|
|
$qc=new Fiche($this->db);
|
|
$qc->get_by_qcode(${'qc_'.$i}, false);
|
|
$sposte=$qc->strAttribut(ATTR_DEF_ACCOUNT);
|
|
/* if there are 2 accounts take following the deb or cred */
|
|
if (strpos($sposte, ',')!=0)
|
|
{
|
|
$array=explode(",", $sposte);
|
|
$poste=(isset(${'ck'.$i}))?$array[0]:$array[1];
|
|
}
|
|
else
|
|
{
|
|
$poste=$sposte;
|
|
if ($poste=='')
|
|
throw new Exception(sprintf(_("La fiche %s n'a pas de poste comptable"),
|
|
${"qc_".$i}));
|
|
}
|
|
$quick_code=${'qc_'.$i};
|
|
}
|
|
else
|
|
{
|
|
$poste=${'poste'.$i};
|
|
}
|
|
|
|
$acc_op->date=$e_date;
|
|
// compute the periode is do not check it
|
|
if ($check_periode==false)
|
|
$acc_op->periode=$oPeriode->p_id;
|
|
$acc_op->desc=null;
|
|
if (strlen(trim(${'ld'.$i}))!=0)
|
|
$acc_op->desc=${'ld'.$i};
|
|
$acc_op->amount=round(${'amount'.$i}, 2);
|
|
$acc_op->grpt=$seq;
|
|
$acc_op->poste=$poste;
|
|
$acc_op->jrn=$this->id;
|
|
$acc_op->type=(isset(${'ck'.$i}))?'d':'c';
|
|
$acc_op->qcode=$quick_code;
|
|
$j_id=$acc_op->insert_jrnx();
|
|
$tot_amount+=round($acc_op->amount, 2);
|
|
$tot_deb+=($acc_op->type=='d')?$acc_op->amount:0;
|
|
$tot_cred+=($acc_op->type=='c')?$acc_op->amount:0;
|
|
if ($g_parameter->MY_ANALYTIC!="nu")
|
|
{
|
|
if (preg_match("/^[6,7]+/", $poste)==1)
|
|
{
|
|
|
|
// for each item, insert into operation_analytique */
|
|
$op=new Anc_Operation($this->db);
|
|
$op->oa_group=$group;
|
|
$op->j_id=$j_id;
|
|
$op->oa_date=$e_date;
|
|
$op->oa_debit=($acc_op->type=='d' )?'t':'f';
|
|
$op->oa_description=$desc;
|
|
$op->save_form_plan($p_array, $count, $j_id);
|
|
$count++;
|
|
}
|
|
}
|
|
}// loop for each item
|
|
$acc_end=new Acc_Operation($this->db);
|
|
$acc_end->amount=$tot_deb;
|
|
if ($check_periode==false)
|
|
$acc_end->periode=$oPeriode->p_id;
|
|
$acc_end->date=$e_date;
|
|
$acc_end->desc=$desc;
|
|
$acc_end->grpt=$seq;
|
|
$acc_end->jrn=$this->id;
|
|
$acc_end->mt=$mt;
|
|
$acc_end->jr_optype=$jr_optype;
|
|
$jr_id=$acc_end->insert_jrn();
|
|
$this->jr_id=$jr_id;
|
|
if ($jr_id==false)
|
|
throw new Exception(_('Balance incorrecte'));
|
|
$acc_end->pj=$e_pj;
|
|
|
|
/* if e_suggest != e_pj then do not increment sequence */
|
|
if (strcmp($e_pj, $e_pj_suggest)==0&&strlen(trim($e_pj))!=0)
|
|
{
|
|
$this->inc_seq_pj();
|
|
}
|
|
|
|
$this->pj=$acc_end->set_pj();
|
|
|
|
$this->db->exec_sql("update jrn set jr_internal='".$internal."' where ".
|
|
" jr_grpt_id = ".$seq);
|
|
$this->internal=$internal;
|
|
// Save now the predef op
|
|
//------------------------
|
|
if (isset($opd_name)&&trim($opd_name)!="")
|
|
{
|
|
$opd=new Pre_Op_Advanced($this->db);
|
|
$opd->get_post();
|
|
$opd->save();
|
|
}
|
|
|
|
if (isset($this->with_concerned)&&$this->with_concerned==true)
|
|
{
|
|
$orap=new acc_reconciliation($this->db);
|
|
$orap->jr_id=$jr_id;
|
|
|
|
$orap->insert($jrn_concerned);
|
|
}
|
|
/**
|
|
* Save the file is any
|
|
*/
|
|
if (isset($_FILES["pj"]))
|
|
{
|
|
$this->db->save_receipt($seq);
|
|
}
|
|
}
|
|
catch (Exception $e)
|
|
{
|
|
record_log($e->getTraceAsString());
|
|
$this->db->rollback();
|
|
echo _('OPERATION ANNULEE ');
|
|
echo '<hr>';
|
|
echo __FILE__.__LINE__.$e->getMessage();
|
|
exit();
|
|
}
|
|
$this->db->commit();
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* @brief get all the data from request and build the object
|
|
*/
|
|
function get_request()
|
|
{
|
|
$this->id=$_REQUEST['p_jrn'];
|
|
}
|
|
|
|
/**
|
|
* @brief retrieve the next number for this type of ledger
|
|
* @param p_cn connx
|
|
* @param p_type ledger type
|
|
*
|
|
* \return the number
|
|
*
|
|
*
|
|
*/
|
|
static function next_number($p_cn, $p_type)
|
|
{
|
|
|
|
$Ret=$p_cn->count_sql("select * from jrn_def where jrn_def_type='".$p_type."'");
|
|
return $Ret+1;
|
|
}
|
|
|
|
/**
|
|
* @brief get the first ledger
|
|
* @param type
|
|
* @return the j_id
|
|
*/
|
|
public function get_first($p_type, $p_access=3)
|
|
{
|
|
global $g_user;
|
|
$all=$g_user->get_ledger($p_type, $p_access);
|
|
return $all[0];
|
|
}
|
|
|
|
/**
|
|
* @brief Update the paiment in the list of operation
|
|
* @param $p_array is normally $_GET
|
|
*/
|
|
function update_paid($p_array)
|
|
{
|
|
// reset all the paid flag because the checkbox is post only
|
|
// when checked
|
|
foreach ($p_array as $name=> $paid)
|
|
{
|
|
list($ad)=sscanf($name, "set_jr_id%d");
|
|
if ($ad==null)
|
|
continue;
|
|
$sql="update jrn set jr_rapt='' where jr_id=$ad";
|
|
$Res=$this->db->exec_sql($sql);
|
|
}
|
|
// set a paid flag for the checked box
|
|
foreach ($p_array as $name=> $paid)
|
|
{
|
|
list ($id)=sscanf($name, "rd_paid%d");
|
|
if ($id==null)
|
|
continue;
|
|
|
|
$sql="update jrn set jr_rapt='paid' where jr_id=$id";
|
|
$Res=$this->db->exec_sql($sql);
|
|
}
|
|
}
|
|
|
|
function update_internal_code($p_internal)
|
|
{
|
|
if (!isset($this->grpt_id))
|
|
throw new Exception(('ERREUR '.__FILE__.":".__LINE__));
|
|
$Res=$this->db->exec_sql("update jrn set jr_internal='".$p_internal."' where ".
|
|
" jr_grpt_id = ".$this->grpt_id);
|
|
}
|
|
|
|
/**
|
|
* Return an array of default card for the ledger type given
|
|
*
|
|
* @param $p_ledger_type VEN ACH ODS or FIN
|
|
* @param $p_side D for Debit or C for credit or NA No Applicable
|
|
*/
|
|
function get_default_card($p_ledger_type, $p_side)
|
|
{
|
|
$array=array();
|
|
$fiche_def_ref=new Fiche_Def_Ref($this->db);
|
|
// ----- for FINANCIAL ----
|
|
if ($p_ledger_type=='FIN')
|
|
{
|
|
$array=$fiche_def_ref->get_by_modele(FICHE_TYPE_CLIENT);
|
|
$array=array_merge($array,
|
|
$fiche_def_ref->get_by_modele(FICHE_TYPE_FOURNISSEUR));
|
|
$array=array_merge($array,
|
|
$fiche_def_ref->get_by_modele(FICHE_TYPE_FIN));
|
|
$array=array_merge($array,
|
|
$fiche_def_ref->get_by_modele(FICHE_TYPE_ADM_TAX));
|
|
$array=array_merge($array,
|
|
$fiche_def_ref->get_by_modele(FICHE_TYPE_EMPL));
|
|
}
|
|
// --- for miscellaneous ----
|
|
if ($p_ledger_type=='ODS')
|
|
{
|
|
$result=$this->db->get_array('select fd_id from fiche_def');
|
|
for ($i=0; $i<count($result); $i++)
|
|
{
|
|
$array[$i]=$result[$i]['fd_id'];
|
|
}
|
|
}
|
|
if ($p_side=='D')
|
|
{
|
|
switch ($p_ledger_type)
|
|
{
|
|
case 'VEN':
|
|
$array=$fiche_def_ref->get_by_modele(FICHE_TYPE_CLIENT);
|
|
break;
|
|
case 'ACH':
|
|
$array=$fiche_def_ref->get_by_modele(FICHE_TYPE_ACH_SER);
|
|
$array=array_merge($array,
|
|
$fiche_def_ref->get_by_modele(FICHE_TYPE_ACH_MAR));
|
|
$array=array_merge($array,
|
|
$fiche_def_ref->get_by_modele(FICHE_TYPE_ACH_MAT));
|
|
break;
|
|
default :
|
|
throw new Exception(_('get_default_card p_ledger_side is invalide ['.$p_ledger_type.']'));
|
|
}
|
|
}
|
|
elseif ($p_side=='C')
|
|
{
|
|
switch ($p_ledger_type)
|
|
{
|
|
case 'VEN':
|
|
$array=$fiche_def_ref->get_by_modele(FICHE_TYPE_VENTE);
|
|
break;
|
|
case 'ACH':
|
|
$array=array_merge($array,
|
|
$fiche_def_ref->get_by_modele(FICHE_TYPE_ADM_TAX));
|
|
$array=array_merge($array,
|
|
$fiche_def_ref->get_by_modele(FICHE_TYPE_FOURNISSEUR));
|
|
break;
|
|
default :
|
|
throw new Exception(_('get_default_card p_ledger_side is invalide ['.$p_ledger_type.']'));
|
|
}
|
|
}
|
|
return $array;
|
|
/*
|
|
$return=array();
|
|
$return = array_values($array);
|
|
for ($i = 0;$i<count($array);$i++ )
|
|
{
|
|
$return[$i]=$array[$i]['fd_id'];
|
|
}
|
|
return $return;
|
|
*
|
|
*/
|
|
}
|
|
|
|
/**
|
|
* @brief retrieve all the card for this type of ledger, make them
|
|
* into a string separated by comma
|
|
* @return all the card or null is nothing is found
|
|
*/
|
|
function get_all_fiche_def()
|
|
{
|
|
$sql="select jrn_def_fiche_deb as deb,jrn_def_fiche_cred as cred ".
|
|
" from jrn_def where ".
|
|
" jrn_def_id = $1 ";
|
|
|
|
$r=$this->db->exec_sql($sql, array($this->id));
|
|
|
|
$res=Database::fetch_all($r);
|
|
if (empty($res))
|
|
return null;
|
|
$card="";
|
|
$comma='';
|
|
foreach ($res as $item)
|
|
{
|
|
if (strlen(trim($item['deb']))!=0)
|
|
{
|
|
$card.=$comma.$item['deb'];
|
|
$comma=',';
|
|
}
|
|
if (strlen(trim($item['cred']))!='')
|
|
{
|
|
$card.=$comma.$item['cred'];
|
|
$comma=',';
|
|
}
|
|
}
|
|
|
|
return $card;
|
|
}
|
|
|
|
/**
|
|
* @brief get the saldo of an exercice, used for the opening of a folder
|
|
* @param$p_exercice is the exercice we want
|
|
* \return an array
|
|
* index =
|
|
* - solde (debit > 0 ; credit < 0)
|
|
* - j_poste
|
|
* - j_qcode
|
|
*/
|
|
function get_saldo_exercice($p_exercice)
|
|
{
|
|
$sql="select sum(a.montant) as solde, j_poste, j_qcode
|
|
from
|
|
(select j_id, case when j_debit='t' then j_montant
|
|
else j_montant * (-1) end as montant
|
|
from jrnx) as a
|
|
join jrnx using (j_id)
|
|
join parm_periode on (j_tech_per = p_id )
|
|
where
|
|
p_exercice=$1
|
|
and j_poste::text not like '7%'
|
|
and j_poste::text not like '6%'
|
|
group by j_poste,j_qcode
|
|
having (sum(a.montant) != 0 ) order by 1 desc";
|
|
$res=$this->db->get_array($sql, array($p_exercice));
|
|
return $res;
|
|
}
|
|
|
|
/**
|
|
* @brief Check if a Dossier is using the strict mode or not
|
|
* \return true if we are using the strict_mode
|
|
*/
|
|
function check_strict()
|
|
{
|
|
global $g_parameter;
|
|
if ($g_parameter->MY_STRICT=='Y')
|
|
return true;
|
|
if ($g_parameter->MY_STRICT=='N')
|
|
return false;
|
|
throw new Exception("Valeur invalid ".__FILE__.':'.__LINE__);
|
|
}
|
|
|
|
/**
|
|
* @brief Check if a Dossier is using the check on the periode, if true than the user has to enter the date
|
|
* and the periode, it is a security check
|
|
* \return true if we are using the double encoding (date+periode)
|
|
*/
|
|
function check_periode()
|
|
{
|
|
global $g_parameter;
|
|
if ($g_parameter->MY_CHECK_PERIODE=='Y')
|
|
return true;
|
|
if ($g_parameter->MY_CHECK_PERIODE=='N')
|
|
return false;
|
|
throw new Exception("Valeur invalid ".__FILE__.':'.__LINE__);
|
|
}
|
|
|
|
/**
|
|
* @brief get the date of the last operation
|
|
*/
|
|
function get_last_date()
|
|
{
|
|
if ($this->id==0)
|
|
throw new Exception(__FILE__.":".__LINE__."Journal incorrect ");
|
|
$sql="select to_char(max(jr_date),'DD.MM.YYYY') from jrn where jr_def_id=$1";
|
|
$date=$this->db->get_value($sql, array($this->id));
|
|
return $date;
|
|
}
|
|
|
|
/**
|
|
* @brief retrieve the jr_id thanks the internal code, do not change
|
|
* anything to the current object
|
|
* @param internal code
|
|
* \return the jr_id or 0 if not found
|
|
*/
|
|
function get_id($p_internal)
|
|
{
|
|
$sql='select jr_id from jrn where jr_internal=$1';
|
|
$value=$this->db->get_value($sql, array($p_internal));
|
|
if ($value=='')
|
|
$value=0;
|
|
return $value;
|
|
}
|
|
|
|
/**
|
|
* @brief create the invoice and saved it as attachment to the
|
|
* operation,
|
|
* @param $internal is the internal code
|
|
* @param $p_array is normally the $_POST
|
|
* \return a string
|
|
*/
|
|
function create_document($internal, $p_array)
|
|
{
|
|
extract($p_array, EXTR_SKIP);
|
|
$doc=new Document($this->db);
|
|
$doc->f_id=$e_client;
|
|
$doc->md_id=$gen_doc;
|
|
$doc->ag_id=0;
|
|
$p_array['e_pj']=$this->pj;
|
|
$filename="";
|
|
$doc->Generate($p_array, $p_array['e_pj']);
|
|
// Move the document to the jrn
|
|
$doc->MoveDocumentPj($internal);
|
|
// Update the comment with invoice number, if the comment is empty
|
|
if (!isset($e_comm)||strlen(trim($e_comm))==0)
|
|
{
|
|
$sql="update jrn set jr_comment=' document ".$doc->d_number."' where jr_internal='$internal'";
|
|
$this->db->exec_sql($sql);
|
|
}
|
|
return h($doc->d_name.' ('.$doc->d_filename.')');
|
|
}
|
|
|
|
/**
|
|
* @brief check if the payment method is valid
|
|
* @param $e_mp is the value and $e_mp_qcode is the quickcode
|
|
* \return nothing throw an Exception
|
|
*/
|
|
public function check_payment($e_mp, $e_mp_qcode)
|
|
{
|
|
/* Check if the "paid by" is empty, */
|
|
if ($e_mp!=0)
|
|
{
|
|
/* the paid by is not empty then check if valid */
|
|
$empl=new Fiche($this->db);
|
|
$empl->get_by_qcode($e_mp_qcode);
|
|
if ($empl->empty_attribute(ATTR_DEF_ACCOUNT)==true)
|
|
{
|
|
throw new Exception(_("Celui qui paie n' a pas de poste comptable"),
|
|
20);
|
|
}
|
|
/* get the account and explode if necessary */
|
|
$sposte=$empl->strAttribut(ATTR_DEF_ACCOUNT);
|
|
// if 2 accounts, take only the debit one for customer
|
|
if (strpos($sposte, ',')!=0)
|
|
{
|
|
$array=explode(',', $sposte);
|
|
$poste_val=$array[0];
|
|
}
|
|
else
|
|
{
|
|
$poste_val=$sposte;
|
|
}
|
|
$poste=new Acc_Account_Ledger($this->db, $poste_val);
|
|
if ($poste->load()==false)
|
|
{
|
|
throw new Exception(sprintf(_("Pour la fiche %s le poste comptable [%s] n'existe pas"),
|
|
$empl->quick_code, $poste->id), 9);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @brief increment the sequence for the pj */
|
|
function inc_seq_pj()
|
|
{
|
|
$sql="select nextval('s_jrn_pj".$this->id."')";
|
|
$this->db->exec_sql($sql);
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
* @brief return the last p_limit operation into an array
|
|
* @param$p_limit is the max of operation to return
|
|
* \return $p_array of Follow_Up object
|
|
*/
|
|
function get_last($p_limit)
|
|
{
|
|
global $g_user;
|
|
$filter_ledger=$g_user->get_ledger_sql('ALL', 3);
|
|
$filter_ledger=str_replace('jrn_def_id', 'jr_def_id', $filter_ledger);
|
|
$sql="
|
|
select jr_id,jr_pj_number,jr_date,to_char(jr_date,'DD.MM.YYYY') as jr_date_fmt,jr_montant, jr_comment,jr_internal,jrn_def_code
|
|
from jrn
|
|
join jrn_def on (jrn_def_id=jr_def_id)
|
|
where $filter_ledger
|
|
order by jr_date desc , substring(jr_pj_number,'[0-9]+$')::numeric desc limit $p_limit";
|
|
$array=$this->db->get_array($sql);
|
|
return $array;
|
|
}
|
|
|
|
/**
|
|
* @brief retreive the jr_grpt_id from a ledger
|
|
* @param $p_what the column to seek
|
|
* possible values are
|
|
* - internal
|
|
* @param $p_value the value of the col.
|
|
*/
|
|
function search_group($p_what, $p_value)
|
|
{
|
|
switch ($p_what)
|
|
{
|
|
case 'internal':
|
|
return $this->db->get_value('select jr_grpt_id from jrn where jr_internal=$1',
|
|
array($p_value));
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @brief retrieve operation from jrn
|
|
* @param $p_from periode (id)
|
|
* @param $p_to periode (id)
|
|
* @return an array
|
|
*/
|
|
function get_operation($p_from, $p_to)
|
|
{
|
|
global $g_user;
|
|
$jrn=($this->id==0)?'and '.$g_user->get_ledger_sql():' and jr_def_id = '.$this->id;
|
|
$sql="select jr_id as id ,jr_internal as internal, ".
|
|
"jr_pj_number as pj,jr_grpt_id,".
|
|
" to_char(jr_date,'DDMMYY') as date_fmt, ".
|
|
" jr_comment as comment, jr_montant as montant ,".
|
|
" jr_grpt_id,jr_def_id".
|
|
" from jrn join jrn_def on (jr_def_id=jrn_def_id) where ".
|
|
" jr_date >= (select p_start from parm_periode where p_id = $1)
|
|
and jr_date <= (select p_end from parm_periode where p_id = $2)".
|
|
' '.$jrn.' order by jr_date,substring(jr_pj_number,\'[0-9]+$\')::numeric asc';
|
|
$ret=$this->db->get_array($sql, array($p_from, $p_to));
|
|
return $ret;
|
|
}
|
|
|
|
/**
|
|
* @brief return the used VAT code with a rate > 0
|
|
* @return an array of tva_id,tva_label,tva_poste
|
|
*/
|
|
public function existing_vat()
|
|
{
|
|
if ($this->type=='ACH')
|
|
{
|
|
$array=$this->db->get_array("select tva_id,tva_label,tva_poste from tva_rate where tva_rate != 0.0000 ".
|
|
" and exists (select qp_vat_code from quant_purchase
|
|
where qp_vat_code=tva_id and exists (select j_id from jrnx where j_jrn_def = $1)) order by tva_id",
|
|
array($this->id));
|
|
}
|
|
if ($this->type=='VEN')
|
|
{
|
|
$array=$this->db->get_array("select tva_id,tva_label,tva_poste from tva_rate where tva_rate != 0.0000 ".
|
|
" and exists (select qs_vat_code from quant_sold
|
|
where qs_vat_code=tva_id and exists (select j_id from jrnx where j_jrn_def = $1)) order by tva_id",
|
|
array($this->id));
|
|
}
|
|
return $array;
|
|
}
|
|
|
|
/**
|
|
* @brief get the amount of vat for a given jr_grpt_id from the table
|
|
* quant_purchase
|
|
* @param the jr_grpt_id
|
|
* @return array price=htva, [1] = vat,
|
|
* @note
|
|
* @see
|
|
@code
|
|
array
|
|
'price' => string '91.3500' (length=7)
|
|
'vat' => string '0.0000' (length=6)
|
|
'priv' => string '0.0000' (length=6)
|
|
'tva_nd_recup' => string '0.0000' (length=6)
|
|
|
|
@endcode
|
|
*/
|
|
function get_other_amount($p_jr_id)
|
|
{
|
|
if ($this->type=='ACH')
|
|
{
|
|
$array=$this->db->get_array('select sum(qp_price) as price,sum(qp_vat) as vat '.
|
|
',sum(coalesce(qp_nd_amount,0)+coalesce(qp_dep_priv,0)) as priv'.
|
|
',sum(coalesce(qp_nd_tva_recup,0)+coalesce(qp_nd_tva,0)) as tva_nd'.
|
|
',sum(qp_vat_sided) as tva_np'.
|
|
' from quant_purchase join jrnx using(j_id)
|
|
where j_grpt=$1 ', array($p_jr_id));
|
|
$ret=$array[0];
|
|
}
|
|
if ($this->type=='VEN')
|
|
{
|
|
$array=$this->db->get_array('select sum(qs_price) as price,sum(qs_vat) as vat '.
|
|
',0 as priv'.
|
|
',0 as tva_nd'.
|
|
',sum(qs_vat_sided) as tva_np'.
|
|
' from quant_sold join jrnx using(j_id)
|
|
where j_grpt=$1 ', array($p_jr_id));
|
|
$ret=$array[0];
|
|
}
|
|
return $ret;
|
|
}
|
|
|
|
/**
|
|
* @brief get the amount of vat for a given jr_grpt_id from the table
|
|
* quant_purchase
|
|
* @param the jr_grpt_id
|
|
* @return array of sum_vat, tva_label
|
|
* @note
|
|
* @see
|
|
@code
|
|
|
|
@endcode
|
|
*/
|
|
function vat_operation($p_jr_id)
|
|
{
|
|
if ($this->type=='ACH')
|
|
{
|
|
$array=$this->db->get_array('select coalesce(sum(qp_vat),0) as sum_vat,tva_id
|
|
from quant_purchase as p right join tva_rate on (qp_vat_code=tva_id) join jrnx using(j_id)
|
|
where tva_rate !=0.0 and j_grpt=$1 group by tva_id',
|
|
array($p_jr_id));
|
|
}
|
|
if ($this->type=='VEN')
|
|
{
|
|
$array=$this->db->get_array('select coalesce(sum(qs_vat),0) as sum_vat,tva_id
|
|
from quant_sold as p right join tva_rate on (qs_vat_code=tva_id) join jrnx using(j_id)
|
|
where tva_rate !=0.0 and j_grpt=$1 group by tva_id',
|
|
array($p_jr_id));
|
|
}
|
|
return $array;
|
|
}
|
|
|
|
/**
|
|
* @brief retrieve amount of previous periode
|
|
* @param $p_to frmo the start of the exercise until $p_to
|
|
* @return $array with vat, price,other_amount
|
|
* @note
|
|
* @see
|
|
@code
|
|
array
|
|
'price' => string '446.1900' (length=8)
|
|
'vat' => string '21.7600' (length=7)
|
|
'priv' => string '0.0000' (length=6)
|
|
'tva_nd_recup' => string '0.0000' (length=6)
|
|
'tva' =>
|
|
array
|
|
0 =>
|
|
array
|
|
'sum_vat' => string '13.7200' (length=7)
|
|
'tva_id' => string '1' (length=1)
|
|
1 =>
|
|
array
|
|
'sum_vat' => string '8.0400' (length=6)
|
|
'tva_id' => string '3' (length=1)
|
|
2 =>
|
|
array
|
|
'sum_vat' => string '0.0000' (length=6)
|
|
'tva_id' => string '4' (length=1)
|
|
|
|
@endcode
|
|
*/
|
|
function previous_amount($p_to)
|
|
{
|
|
/* get the first periode of exercise */
|
|
$periode=new Periode($this->db, $p_to);
|
|
$exercise=$periode->get_exercice();
|
|
list ($min, $max)=$periode->get_limit($exercise);
|
|
// transform min into date
|
|
$min_date=$min->first_day();
|
|
// transform $p_to into date
|
|
$periode_max=new Periode($this->db, $p_to);
|
|
$max_date=$periode_max->first_day();
|
|
bcscale(2);
|
|
// min periode
|
|
if ($this->type=='ACH')
|
|
{
|
|
/* get all amount exclude vat */
|
|
$sql="select coalesce(sum(qp_price),0) as price".
|
|
" ,coalesce(sum(qp_vat),0) as vat ".
|
|
',coalesce(sum(qp_dep_priv),0) as priv'.
|
|
',coalesce(sum(qp_vat_sided),0) as reversed'.
|
|
',coalesce(sum(qp_nd_tva_recup),0)+coalesce(sum(qp_nd_tva),0) as tva_nd'.
|
|
',coalesce(sum(qp_vat_sided),0) as tva_np'.
|
|
' from quant_purchase join jrnx using(j_id) '.
|
|
" where j_date >= to_date($1,'DD.MM.YYYY') and j_date < to_date($2,'DD.MM.YYYY') ".
|
|
' and j_jrn_def = $3';
|
|
$array=$this->db->get_array($sql,
|
|
array($min_date, $max_date, $this->id));
|
|
|
|
$ret=$array[0];
|
|
/* retrieve all vat code */
|
|
$array=$this->db->get_array("select coalesce(sum(qp_vat),0) as sum_vat,tva_id
|
|
from quant_purchase as p right join tva_rate on (qp_vat_code=tva_id) join jrnx using(j_id)
|
|
where tva_rate !=0 and j_date >= to_date($1,'DD.MM.YYYY') and j_date < to_date($2,'DD.MM.YYYY')
|
|
and j_jrn_def = $3
|
|
group by tva_id",
|
|
array($min_date, $max_date, $this->id));
|
|
$ret['tva']=$array;
|
|
}
|
|
if ($this->type=='VEN')
|
|
{
|
|
/* get all amount exclude vat */
|
|
$sql="select coalesce(sum(qs_price),0) as price".
|
|
" ,coalesce(sum(qs_vat),0) as vat ".
|
|
',0 as priv'.
|
|
',0 as tva_nd'.
|
|
',coalesce(sum(qs_vat_sided),0) as reversed'.
|
|
',coalesce(sum(qs_vat_sided),0) as tva_np'.
|
|
' from quant_sold join jrnx using(j_id) '.
|
|
" where j_date >= to_date($1,'DD.MM.YYYY') and j_date < to_date($2,'DD.MM.YYYY') ".
|
|
' and j_jrn_def = $3';
|
|
$array=$this->db->get_array($sql,
|
|
array($min_date, $max_date, $this->id));
|
|
$ret=$array[0];
|
|
/* retrieve all vat code */
|
|
$array=$this->db->get_array("select coalesce(sum(qs_vat),0) as sum_vat,tva_id
|
|
from quant_sold as p right join tva_rate on (qs_vat_code=tva_id) join jrnx using(j_id)
|
|
where tva_rate !=0 and
|
|
j_date >= to_date($1,'DD.MM.YYYY') and j_date < to_date($2,'DD.MM.YYYY')
|
|
and j_jrn_def = $3
|
|
group by tva_id",
|
|
array($min_date, $max_date, $this->id));
|
|
$ret['tva']=$array;
|
|
}
|
|
if ($this->type=="FIN")
|
|
{
|
|
|
|
/* find the quick code of this ledger */
|
|
$ledger=new Acc_Ledger_Fin($this->db, $this->id);
|
|
$qcode=$ledger->get_bank();
|
|
$bank_card=new Fiche($this->db, $qcode);
|
|
|
|
/* add the amount from Opening Writing */
|
|
$cond=sprintf(" j_jrn_def <> %d and j_date >= to_date('%s','DD.MM.YYYY') and j_date < to_date('%s','DD.MM.YYYY') ",
|
|
$this->id, $min_date, $max_date);
|
|
$saldo=$bank_card->get_bk_balance($cond);
|
|
$ret['amount']=bcsub($saldo['debit'], $saldo['credit']);
|
|
}
|
|
return $ret;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
// TEST MODULE
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
/**
|
|
* @brief this function is intended to test this class
|
|
*/
|
|
static function test_me($pCase='')
|
|
{
|
|
if ($pCase=='')
|
|
{
|
|
echo Acc_Reconciliation::$javascript;
|
|
html_page_start();
|
|
$cn=Dossier::connect();
|
|
$_SESSION['g_user']=NOALYSS_ADMINISTRATOR;
|
|
$_SESSION['g_pass']='phpcompta';
|
|
|
|
$id=(isset($_REQUEST['p_jrn']))?$_REQUEST['p_jrn']:-1;
|
|
$a=new Acc_Ledger($cn, $id);
|
|
$a->with_concerned=true;
|
|
// Vide
|
|
echo '<FORM method="post">';
|
|
echo $a->select_ledger()->input();
|
|
echo HtmlInput::submit('go', 'Test it');
|
|
echo '</form>';
|
|
if (isset($_POST['go']))
|
|
{
|
|
echo "Ok ";
|
|
echo '<form method="post">';
|
|
echo $a->show_form();
|
|
echo HtmlInput::submit('post_id', 'Try me');
|
|
echo '</form>';
|
|
// Show the predef operation
|
|
// Don't forget the p_jrn
|
|
echo '<form>';
|
|
echo dossier::hidden();
|
|
echo '<input type="hidden" value="'.$id.'" name="p_jrn">';
|
|
$op=new Pre_operation($cn);
|
|
$op->p_jrn=$id;
|
|
$op->od_direct='t';
|
|
if ($op->count()!=0)
|
|
{
|
|
echo HtmlInput::submit('use_opd',
|
|
'Utilisez une opération prédéfinie',
|
|
"", "smallbutton");
|
|
echo $op->show_button();
|
|
}
|
|
echo '</form>';
|
|
exit('test_me');
|
|
}
|
|
|
|
if (isset($_POST['post_id']))
|
|
{
|
|
|
|
echo '<form method="post">';
|
|
echo $a->show_form($_POST, 1);
|
|
echo HtmlInput::button('add', 'Ajout d\'une ligne',
|
|
'onClick="quick_writing_add_row()"');
|
|
echo HtmlInput::submit('save_it', _("Sauver"));
|
|
echo '</form>';
|
|
exit('test_me');
|
|
}
|
|
if (isset($_POST['save_it']))
|
|
{
|
|
print 'saving';
|
|
$array=$_POST;
|
|
$array['save_opd']=1;
|
|
try
|
|
{
|
|
$a->save($array);
|
|
}
|
|
catch (Exception $e)
|
|
{
|
|
alert($e->getMessage());
|
|
echo '<form method="post">';
|
|
|
|
echo $a->show_form($_POST);
|
|
echo HtmlInput::submit('post_id', 'Try me');
|
|
echo '</form>';
|
|
}
|
|
return;
|
|
}
|
|
// The GET at the end because automatically repost when you don't
|
|
// specify the url in the METHOD field
|
|
if (isset($_GET['use_opd']))
|
|
{
|
|
$op=new Pre_op_advanced($cn);
|
|
$op->set_od_id($_REQUEST['pre_def']);
|
|
//$op->p_jrn=$id;
|
|
|
|
$p_post=$op->compute_array();
|
|
|
|
echo '<FORM method="post">';
|
|
|
|
echo $a->show_form($p_post);
|
|
echo HtmlInput::submit('post_id', 'Use predefined operation');
|
|
echo '</form>';
|
|
return;
|
|
}
|
|
}// if case = ''
|
|
///////////////////////////////////////////////////////////////////////////
|
|
// search
|
|
if ($pCase=='search')
|
|
{
|
|
html_page_start();
|
|
$cn=Dossier::connect();
|
|
$ledger=new Acc_Ledger($cn, 0);
|
|
$_SESSION['g_user']=NOALYSS_ADMINISTRATOR;
|
|
$_SESSION['g_pass']='phpcompta';
|
|
echo $ledger->search_form('ALL');
|
|
}
|
|
///////////////////////////////////////////////////////////////////////////
|
|
// reverse
|
|
// Give yourself the var and check in your tables
|
|
///////////////////////////////////////////////////////////////////////////
|
|
if ($pCase=='reverse')
|
|
{
|
|
$cn=Dossier::connect();
|
|
$jr_internal='OD-01-272';
|
|
try
|
|
{
|
|
$cn->start();
|
|
$jrn_def_id=$cn->get_value('select jr_def_id from jrn where jr_internal=$1',
|
|
array($jr_internal));
|
|
$ledger=new Acc_Ledger($cn, $jrn_def_id);
|
|
$ledger->jr_id=$cn->get_value('select jr_id from jrn where jr_internal=$1',
|
|
array($jr_internal));
|
|
|
|
echo "Ouvrez le fichier ".__FILE__." à la ligne ".__LINE__." pour changer jr_internal et vérifier le résultat de l'extourne";
|
|
|
|
$ledger->reverse('01.07.2010');
|
|
}
|
|
catch (Exception $e)
|
|
{
|
|
$cn->rollback();
|
|
record_log($e->getTraceAsString());
|
|
}
|
|
$cn->commit();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* create an array of the existing cat, to be used in a checkbox form
|
|
*
|
|
*/
|
|
static function array_cat()
|
|
{
|
|
$r=array(
|
|
array('cat'=>'VEN', 'name'=>_("Journaux de vente")),
|
|
array('cat'=>'ACH', 'name'=>_("Journaux d'achat")),
|
|
array('cat'=>'FIN', 'name'=>_("Journaux Financier")),
|
|
array('cat'=>'ODS', 'name'=>_("Journaux d'Opérations diverses"))
|
|
);
|
|
return $r;
|
|
}
|
|
|
|
//---------------------------------------------------------------------
|
|
/// Return the f_id of the tiers , called by get_tiers
|
|
//!\param $p_jrn_type type of the ledger FIN, VEN ACH or ODS
|
|
//!\param $jr_id jrn.jr_id
|
|
//---------------------------------------------------------------------
|
|
function get_tiers_id($p_jrn_type, $jr_id)
|
|
{
|
|
$tiers=0;
|
|
switch ($p_jrn_type)
|
|
{
|
|
case 'VEN':
|
|
$tiers=$this->db->get_value('select max(qs_client) from quant_sold join jrnx using (j_id) join jrn on (jr_grpt_id=j_grpt) where jrn.jr_id=$1',
|
|
array($jr_id));
|
|
break;
|
|
case 'ACH':
|
|
$tiers=$this->db->get_value('select max(qp_supplier) from quant_purchase join jrnx using (j_id) join jrn on (jr_grpt_id=j_grpt) where jrn.jr_id=$1',
|
|
array($jr_id));
|
|
|
|
break;
|
|
case 'FIN':
|
|
$tiers=$this->db->get_value('select qf_other from quant_fin where jr_id=$1',
|
|
array($jr_id));
|
|
break;
|
|
}
|
|
if ($this->db->count()==0)
|
|
return 0;
|
|
return $tiers;
|
|
}
|
|
|
|
/**
|
|
* Retrieve the third : supplier for purchase, customer for sale, bank for fin,
|
|
* @param $p_jrn_type type of the ledger FIN, VEN ACH or ODS
|
|
* @param $jr_id jrn.jr_id
|
|
*/
|
|
function get_tiers($p_jrn_type, $jr_id)
|
|
{
|
|
if ($p_jrn_type=='ODS')
|
|
return ' ';
|
|
$tiers=$this->get_tiers_id($p_jrn_type, $jr_id);
|
|
if ($tiers==0)
|
|
return "";
|
|
|
|
$name=$this->db->get_value('select ad_value from fiche_detail where ad_id=1 and f_id=$1',
|
|
array($tiers));
|
|
$first_name=$this->db->get_value('select ad_value from fiche_detail where ad_id=32 and f_id=$1',
|
|
array($tiers));
|
|
return $name.' '.$first_name;
|
|
}
|
|
|
|
/**
|
|
* @brief listing of all ledgers
|
|
* @return HTML string
|
|
*/
|
|
function listing()
|
|
{
|
|
$str_dossier=dossier::get();
|
|
$base_url="?".dossier::get()."&ac=".$_REQUEST['ac'];
|
|
|
|
$r="";
|
|
$r.=_('Cherche')." ".HtmlInput::filter_table("cfgledger_table_id", "0",
|
|
"1");
|
|
$r.='<TABLE id="cfgledger_table_id" class="vert_mtitle">';
|
|
$r.='<TR><TD class="first"><A HREF="'.$base_url.'&sa=add">'._('Ajout journal').' </A></TD></TR>';
|
|
$ret=$this->db->exec_sql("select distinct jrn_def_id,jrn_def_name,
|
|
jrn_def_class_deb,jrn_def_class_cred,jrn_def_type
|
|
from jrn_def order by jrn_def_name");
|
|
$Max=Database::num_row($ret);
|
|
|
|
|
|
for ($i=0; $i<$Max; $i++)
|
|
{
|
|
$l_line=Database::fetch_array($ret, $i);
|
|
$url=$base_url."&sa=detail&p_jrn=".$l_line['jrn_def_id'];
|
|
$r.=sprintf('<TR ledger_type="%s"><TD><A HREF="%s">%s</A></TD></TR>', $l_line['jrn_def_type'],$url,
|
|
h($l_line['jrn_def_name']).' ('.$l_line['jrn_def_type'].')');
|
|
}
|
|
$r.="</TABLE>";
|
|
return $r;
|
|
}
|
|
|
|
/**
|
|
* display detail of a ledger
|
|
*
|
|
*/
|
|
function display_ledger()
|
|
{
|
|
if ($this->load()==-1)
|
|
{
|
|
throw new Exception(_("Journal n'existe pas"), -1);
|
|
}
|
|
$type=$this->jrn_def_type;
|
|
$name=$this->jrn_def_name;
|
|
$code=$this->jrn_def_code;
|
|
$str_add_button="";
|
|
/* widget for searching an account */
|
|
$wSearch=new IPoste();
|
|
$wSearch->set_attribute('ipopup', 'ipop_account');
|
|
$wSearch->set_attribute('account', 'p_jrn_class_deb');
|
|
$wSearch->set_attribute('no_overwrite', '1');
|
|
$wSearch->set_attribute('noquery', '1');
|
|
$wSearch->table=3;
|
|
$wSearch->name="p_jrn_class_deb";
|
|
$wSearch->size=20;
|
|
$wSearch->value=$this->jrn_def_class_deb;
|
|
$search=$wSearch->input();
|
|
|
|
$wPjPref=new IText();
|
|
$wPjPref->name='jrn_def_pj_pref';
|
|
$wPjPref->value=$this->jrn_def_pj_pref;
|
|
$pj_pref=$wPjPref->input();
|
|
|
|
$wPjSeq=new INum();
|
|
$wPjSeq->value=0;
|
|
$wPjSeq->name='jrn_def_pj_seq';
|
|
$pj_seq=$wPjSeq->input();
|
|
$last_seq=$this->get_last_pj();
|
|
$name=$this->jrn_def_name;
|
|
|
|
$hidden=HtmlInput::hidden('p_jrn', $this->id);
|
|
$hidden.=HtmlInput::hidden('sa', 'detail');
|
|
$hidden.=dossier::hidden();
|
|
$hidden.=HtmlInput::hidden('p_jrn_deb_max_line', 10);
|
|
$hidden.=HtmlInput::hidden('p_ech_lib', 'echeance');
|
|
$hidden.=HtmlInput::hidden('p_jrn_type', $type);
|
|
|
|
$min_row=new INum("min_row", $this->jrn_deb_max_line);
|
|
$min_row->prec=0;
|
|
|
|
$description=new ITextarea('p_description');
|
|
$description->style='class="itextarea" style="margin:0px;"';
|
|
$description->value=$this->jrn_def_description;
|
|
$str_description=$description->input();
|
|
|
|
/* Load the card */
|
|
$card=$this->get_fiche_def();
|
|
$rdeb=explode(',', $card['deb']);
|
|
$rcred=explode(',', $card['cred']);
|
|
/* Numbering (only FIN) */
|
|
$num_op=new ICheckBox('numb_operation');
|
|
if ($this->jrn_def_num_op==1)
|
|
$num_op->selected=true;
|
|
/* bank card */
|
|
$qcode_bank='';
|
|
if ($type=='FIN')
|
|
{
|
|
$f_id=$this->jrn_def_bank;
|
|
if (isNumber($f_id)==1)
|
|
{
|
|
$fBank=new Fiche($this->db, $f_id);
|
|
$qcode_bank=$fBank->get_quick_code();
|
|
}
|
|
}
|
|
$new=0;
|
|
$cn=$this->db;
|
|
echo $hidden;
|
|
$actif=new ISelect("jrn_enable");
|
|
$actif->value=[
|
|
["label"=>_("Activé"),"value"=>1],
|
|
["label"=>_("Désactivé"),"value"=>0]
|
|
];
|
|
$actif->selected=$this->jrn_enable;
|
|
require_once NOALYSS_TEMPLATE.'/param_jrn.php';
|
|
}
|
|
|
|
/**
|
|
* Verify before update
|
|
*
|
|
* @param type $array
|
|
* 'p_jrn' => string '3' (length=1)
|
|
'sa' => string 'detail' (length=6)
|
|
'gDossier' => string '82' (length=2)
|
|
'p_jrn_deb_max_line' => string '10' (length=2)
|
|
'p_ech_lib' => string 'echeance' (length=8)
|
|
'p_jrn_type' => string 'ACH' (length=3)
|
|
'p_jrn_name' => string 'Achat' (length=5)
|
|
'jrn_def_pj_pref' => string 'ACH' (length=3)
|
|
'jrn_def_pj_seq' => string '0' (length=1)
|
|
'FICHECRED' =>
|
|
array
|
|
0 => string '4' (length=1)
|
|
'FICHEDEB' =>
|
|
array
|
|
0 => string '7' (length=1)
|
|
1 => string '5' (length=1)
|
|
2 => string '13' (length=2)
|
|
'update' => string 'Sauve' (length=5
|
|
* @exception is throw is test are not valid
|
|
*/
|
|
function verify_ledger($array)
|
|
{
|
|
extract($array, EXTR_SKIP);
|
|
try
|
|
{
|
|
if (isNumber($p_jrn)==0)
|
|
throw new Exception("Id invalide");
|
|
if (isNumber($p_jrn_deb_max_line)==0)
|
|
throw new Exception(_("Nombre de ligne incorrect"));
|
|
if (trim($p_jrn_name)=="")
|
|
throw new Exception("Nom de journal invalide");
|
|
if ($this->db->get_value("select count(*) from jrn_def where jrn_def_name=$1 and jrn_Def_id<>$2",
|
|
array($p_jrn_name, $p_jrn))>0)
|
|
throw new Exception(_("Un journal avec ce nom existe déjà"));
|
|
if ($p_jrn_type=='FIN')
|
|
{
|
|
$a=new Fiche($this->db);
|
|
$result=$a->get_by_qcode(trim(strtoupper($_POST['bank'])), false);
|
|
if ($result==1)
|
|
throw new Exception(_("Aucun compte en banque n'est donné"));
|
|
}
|
|
if ($p_jrn_type=="-1")
|
|
{
|
|
throw new Exception(_('Choix du type de journal est obligatoire'));
|
|
}
|
|
}
|
|
catch (Exception $e)
|
|
{
|
|
record_log($e->getTraceAsString());
|
|
throw $e;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* update a ledger
|
|
* @param type $array normally post
|
|
* @see verify_ledger
|
|
*/
|
|
function update($array='')
|
|
{
|
|
if ($array==null)
|
|
throw new Exception('save cannot use a empty array');
|
|
|
|
extract($array, EXTR_SKIP);
|
|
$this->jrn_def_id=$p_jrn;
|
|
$this->jrn_def_name=$p_jrn_name;
|
|
$this->jrn_def_ech_lib=$p_ech_lib;
|
|
$this->jrn_def_max_line_deb=($p_jrn_deb_max_line<1)?1:$p_jrn_deb_max_line;
|
|
$this->jrn_def_type=$p_jrn_type;
|
|
$this->jrn_def_pj_pref=$jrn_def_pj_pref;
|
|
$this->jrn_deb_max_line=($min_row<1)?1:$min_row;
|
|
$this->jrn_def_description=$p_description;
|
|
$this->jrn_enable=$jrn_enable;
|
|
switch ($this->jrn_def_type)
|
|
{
|
|
case 'ACH':
|
|
$this->jrn_def_fiche_cred=(isset($ACH_FICHECRED))?join($ACH_FICHECRED,
|
|
','):'';
|
|
$this->jrn_def_fiche_deb=(isset($ACH_FICHEDEB))?join($ACH_FICHEDEB,
|
|
','):"";
|
|
break;
|
|
case 'VEN':
|
|
$this->jrn_def_fiche_cred=(isset($VEN_FICHECRED))?join($VEN_FICHECRED,
|
|
','):'';
|
|
$this->jrn_def_fiche_deb=(isset($VEN_FICHEDEB))?join($VEN_FICHEDEB,
|
|
','):"";
|
|
|
|
break;
|
|
case 'ODS':
|
|
$this->jrn_def_class_deb=$p_jrn_class_deb;
|
|
$this->jrn_def_fiche_deb=(isset($ODS_FICHEDEB))?join($ODS_FICHEDEB,
|
|
','):''; ;
|
|
$this->jrn_def_fiche_cred=null;
|
|
break;
|
|
|
|
case 'FIN':
|
|
$a=new Fiche($this->db);
|
|
$result=$a->get_by_qcode(trim(strtoupper($_POST['bank'])), false);
|
|
$bank=$a->id;
|
|
$this->jrn_def_bank=$bank;
|
|
$this->jrn_def_fiche_deb=(isset($FIN_FICHEDEB))?join($FIN_FICHEDEB,
|
|
','):"";
|
|
if ($result==-1)
|
|
throw new Exception(_("Aucun compte en banque n'est donné"));
|
|
$this->jrn_def_num_op=(isset($numb_operation))?1:0;
|
|
break;
|
|
}
|
|
|
|
parent::update();
|
|
//Reset sequence if needed
|
|
if ($jrn_def_pj_seq!=0)
|
|
{
|
|
$Res=$this->db->alter_seq("s_jrn_pj".$p_jrn, $jrn_def_pj_seq);
|
|
}
|
|
}
|
|
/**
|
|
* Create the section payment
|
|
* @param int $p_selected
|
|
* @param number $p_amount
|
|
* @param date $p_date or empty string
|
|
* @param string $p_comm or empty comm
|
|
* @return string
|
|
*/
|
|
function input_paid($p_selected,$p_amount=0,$p_date="",$p_comm="")
|
|
{
|
|
$r='';
|
|
$r.='<div id="payment"> ';
|
|
$r.='<h2> '._('Paiement').' </h2>';
|
|
$mp=new Acc_Payment($this->db);
|
|
$mp->set_parameter('ledger_source', $this->id);
|
|
$r.=$mp->select($p_selected,$p_amount,$p_date,$p_comm);
|
|
$r.='</div>';
|
|
return $r;
|
|
}
|
|
|
|
/**
|
|
* display screen to enter a new ledger
|
|
*/
|
|
function input_new()
|
|
{
|
|
$http=new HttpInput();
|
|
$retry=$http->post("sa", "string", "");
|
|
// if ( $retry == "add") {
|
|
$default_type=$http->post("p_jrn_type", "string", -1);
|
|
$previous_jrn_def_pj_pref=$http->post("jrn_def_pj_pref", "string", "");
|
|
$previous_p_description=$http->post("p_description", "string", "");
|
|
$previous_p_jrn_name=$http->post('p_jrn_name', "string", '');
|
|
$previous_p_jrn_type=$http->post("p_jrn_type", "string", "");
|
|
// }
|
|
global $g_user;
|
|
$f_add_button=new ISmallButton('add_card');
|
|
$f_add_button->label=_('Créer une nouvelle fiche');
|
|
$f_add_button->tabindex=-1;
|
|
$f_add_button->set_attribute('jrn', -1);
|
|
$f_add_button->javascript=" this.jrn=-1;select_card_type({type_cat:4});";
|
|
|
|
$str_add_button="";
|
|
if ($g_user->check_action(FICADD)==1)
|
|
{
|
|
$str_add_button=$f_add_button->input();
|
|
}
|
|
$wSearch=new IPoste();
|
|
$wSearch->table=3;
|
|
$wSearch->set_attribute('ipopup', 'ipop_account');
|
|
$wSearch->set_attribute('account', 'p_jrn_class_deb');
|
|
$wSearch->set_attribute('no_overwrite', '1');
|
|
$wSearch->set_attribute('noquery', '1');
|
|
|
|
$wSearch->name="p_jrn_class_deb";
|
|
$wSearch->size=20;
|
|
|
|
$search=$wSearch->input();
|
|
// default for ACH
|
|
$default_deb_purchase=$this->get_default_card('ACH', 'D');
|
|
$default_cred_purchase=$this->get_default_card('ACH', 'C');
|
|
|
|
// default for VEN
|
|
$default_deb_sale=$this->get_default_card('VEN', 'D');
|
|
$default_cred_sale=$this->get_default_card('VEN', 'C');
|
|
|
|
// default for FIN
|
|
$default_fin=$this->get_default_card("FIN", "");
|
|
|
|
//default ods
|
|
$default_ods=$this->get_default_card("ODS", "");
|
|
|
|
/* construct all the hidden */
|
|
$hidden=HtmlInput::hidden('p_jrn', -1);
|
|
$hidden.=HtmlInput::hidden('p_action', 'jrn');
|
|
$hidden.=HtmlInput::hidden('sa', 'add');
|
|
$hidden.=dossier::hidden();
|
|
$hidden.=HtmlInput::hidden('p_jrn_deb_max_line', 10);
|
|
$hidden.=HtmlInput::hidden('p_ech_lib', 'echeance');
|
|
|
|
/* properties of the ledger */
|
|
$name=$previous_p_jrn_name;
|
|
$code="";
|
|
$wType=new ISelect();
|
|
$a_jrn=$this->db->make_array("select '-1',' -- "._("choix du type de journal")." -- ' union select jrn_type_id,jrn_desc from jrn_type");
|
|
$wType->selected='-1';
|
|
$wType->value=$a_jrn;
|
|
$wType->name="p_jrn_type";
|
|
$wType->id="p_jrn_type_select_id";
|
|
$wType->javascript=' onchange="show_ledger_div()"';
|
|
$wType->selected=$default_type;
|
|
$type=$wType->input();
|
|
$rcred=$rdeb=array();
|
|
$wPjPref=new IText();
|
|
$wPjPref->name='jrn_def_pj_pref';
|
|
$wPjPref->value=$previous_jrn_def_pj_pref;
|
|
$pj_pref=$wPjPref->input();
|
|
$pj_seq='';
|
|
$last_seq=0;
|
|
$new=1;
|
|
$description=new ITextarea('p_description');
|
|
$description->style='class="itextarea" style="margin:0px;"';
|
|
$description->value=$previous_p_description;
|
|
$str_description=$description->input();
|
|
/* bank card */
|
|
$qcode_bank='';
|
|
/* Numbering (only FIN) */
|
|
$num_op=new ICheckBox('numb_operation');
|
|
echo dossier::hidden();
|
|
echo HtmlInput::hidden('ac', $_REQUEST['ac']);
|
|
echo $hidden;
|
|
|
|
$cn=$this->db;
|
|
$min_row=new INum("min_row", MAX_ARTICLE);
|
|
$min_row->prec=0;
|
|
require_once NOALYSS_TEMPLATE.'/param_jrn.php';
|
|
}
|
|
|
|
/**
|
|
* Insert a new ledger
|
|
* @param type $array normally $_POST
|
|
* @see verify_ledger
|
|
*/
|
|
function save_new($array)
|
|
{
|
|
$this->load();
|
|
extract($array, EXTR_SKIP);
|
|
$this->jrn_def_id=-1;
|
|
$this->jrn_def_name=$p_jrn_name;
|
|
$this->jrn_def_ech_lib=$p_ech_lib;
|
|
$this->jrn_def_max_line_deb=$p_jrn_deb_max_line;
|
|
$this->jrn_def_type=$p_jrn_type;
|
|
$this->jrn_def_pj_pref=$jrn_def_pj_pref;
|
|
$this->jrn_deb_max_line=$min_row;
|
|
$this->jrn_def_code=sprintf("%s%02d",
|
|
trim(substr($this->jrn_def_type, 0, 1)),
|
|
Acc_Ledger::next_number($this->db, $this->jrn_def_type));
|
|
$this->jrn_def_description=$p_description;
|
|
switch ($this->jrn_def_type)
|
|
{
|
|
case 'ACH':
|
|
$this->jrn_def_fiche_cred=(isset($ACH_FICHECRED))?join($ACH_FICHECRED,
|
|
','):'';
|
|
$this->jrn_def_fiche_deb=(isset($ACH_FICHEDEB))?join($ACH_FICHEDEB,
|
|
','):"";
|
|
break;
|
|
case 'VEN':
|
|
$this->jrn_def_fiche_cred=(isset($VEN_FICHECRED))?join($VEN_FICHECRED,
|
|
','):'';
|
|
$this->jrn_def_fiche_deb=(isset($VEN_FICHEDEB))?join($VEN_FICHEDEB,
|
|
','):"";
|
|
|
|
break;
|
|
case 'ODS':
|
|
$this->jrn_def_class_deb=$p_jrn_class_deb;
|
|
$this->jrn_def_fiche_deb=(isset($ODS_FICHEDEB))?join($ODS_FICHEDEB,
|
|
','):''; ;
|
|
$this->jrn_def_fiche_cred=null;
|
|
break;
|
|
case 'FIN':
|
|
$a=new Fiche($this->db);
|
|
$result=$a->get_by_qcode(trim(strtoupper($_POST['bank'])), false);
|
|
$bank=$a->id;
|
|
$this->jrn_def_bank=$bank;
|
|
$this->jrn_def_fiche_deb=(isset($FIN_FICHEDEB))?join($FIN_FICHEDEB,
|
|
','):"";
|
|
if ($result==-1)
|
|
throw new Exception(_("Aucun compte en banque n'est donné"));
|
|
$this->jrn_def_num_op=(isset($numb_operation))?1:0;
|
|
break;
|
|
}
|
|
|
|
parent::insert();
|
|
}
|
|
|
|
/**
|
|
* delete a ledger IF is not already used
|
|
* @exception : cannot delete
|
|
*/
|
|
function delete_ledger()
|
|
{
|
|
try
|
|
{
|
|
if ($this->db->get_value("select count(jr_id) from jrn where jr_def_id=$1",
|
|
array($this->jrn_def_id))>0)
|
|
throw new Exception(_("Impossible d'effacer un journal qui contient des opérations"));
|
|
parent::delete();
|
|
}
|
|
catch (Exception $e)
|
|
{
|
|
record_log($e->getTraceAsString());
|
|
throw $e;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get operation from the ledger type before, after or with the
|
|
* given date . The array is filtered by the ledgers granted to the
|
|
* user
|
|
* @global type $g_user
|
|
* @param $p_date Date (d.m.Y)
|
|
* @param $p_ledger_type VEN ACH
|
|
* @param type $sql_op < > or =
|
|
* @return array from jrn (jr_id, jr_internal, jr_date, jr_comment,jr_pj_number,jr_montant)
|
|
* @throws Exception
|
|
*/
|
|
function get_operation_date($p_date, $p_ledger_type, $sql_op)
|
|
{
|
|
global $g_user;
|
|
switch ($p_ledger_type)
|
|
{
|
|
case 'ACH':
|
|
$filter=$g_user->get_ledger_sql('ACH', 3);
|
|
break;
|
|
case 'VEN':
|
|
$filter=$g_user->get_ledger_sql('VEN', 3);
|
|
break;
|
|
default:
|
|
throw new Exception('Ledger_type invalid : '.$p_ledger_type);
|
|
}
|
|
|
|
|
|
$sql="select jr_id, jr_internal, jr_date, jr_comment,jr_pj_number,jr_montant
|
|
from jrn
|
|
join jrn_def on (jrn_def_id=jr_def_id)
|
|
where
|
|
jr_ech is not null
|
|
and jr_ech $sql_op to_date($1,'DD.MM.YYYY')
|
|
and coalesce (jr_rapt,'xx') <> 'paid'
|
|
and $filter
|
|
";
|
|
$array=$this->db->get_array($sql, array($p_date));
|
|
return $array;
|
|
}
|
|
/** @brief Get simplified row from ledger
|
|
* Call Acc_Ledger_History_Generic:get_rowSimple
|
|
* @param p_from periode
|
|
* @param p_to periode
|
|
* @param p_limit starting line
|
|
* @param p_offset number of lines
|
|
* @param trunc if data must be truncated (pdf export)
|
|
*
|
|
* \return an Array with the asked data
|
|
*/
|
|
function get_rowSimple($p_from, $p_to, $pa_ledger=[],$trunc=0,$p_limit=-1,$p_offset=-1)
|
|
{
|
|
if ( empty($pa_ledger) ) {
|
|
$pa_ledger=[$this->id];
|
|
}
|
|
// if $pa_ledger == 0, it means we need to show all ledgers
|
|
if ( $pa_ledger == [0] ) {
|
|
$pa_ledger=Print_Ledger::available_ledger($p_from);
|
|
}
|
|
|
|
$alh_generic=new Acc_Ledger_History_Generic($this->db, $pa_ledger, $p_from, $p_to, "A");
|
|
$alh_generic->get_rowSimple($trunc,$p_limit,$p_offset);
|
|
$data=$alh_generic->get_data();
|
|
return $data;
|
|
}
|
|
/**
|
|
* @brief get info from supplier to pay today
|
|
*/
|
|
function get_supplier_now()
|
|
{
|
|
$array=$this->get_operation_date(Date('d.m.Y'), 'ACH', '=');
|
|
return $array;
|
|
}
|
|
|
|
/**
|
|
* @brief get info from supplier not yet paid
|
|
*/
|
|
function get_supplier_late()
|
|
{
|
|
$array=$this->get_operation_date(Date('d.m.Y'), 'ACH', '<');
|
|
return $array;
|
|
}
|
|
|
|
/**
|
|
* @brief get info from customer to pay today
|
|
*/
|
|
function get_customer_now()
|
|
{
|
|
$array=$this->get_operation_date(Date('d.m.Y'), 'VEN', '=');
|
|
return $array;
|
|
}
|
|
|
|
/**
|
|
* @brief get info from customer not yet paid
|
|
*/
|
|
function get_customer_late()
|
|
{
|
|
$array=$this->get_operation_date(Date('d.m.Y'), 'VEN', '<');
|
|
return $array;
|
|
}
|
|
|
|
function convert_from_follow($p_ag_id)
|
|
{
|
|
global $g_user;
|
|
if (isNumber($p_ag_id)==0)
|
|
return null;
|
|
if (!$g_user->can_read_action($p_ag_id))
|
|
die(_('Action non accessible'));
|
|
$array=array();
|
|
|
|
// retrieve info from action_gestion
|
|
$tiers_id=$this->db->get_value('select f_id_dest from action_gestion where ag_id=$1',
|
|
array($p_ag_id));
|
|
if ($this->db->size()!=0)
|
|
$qcode=$this->db->get_value('select j_qcode from vw_poste_qcode where f_id=$1',
|
|
array($tiers_id));
|
|
else
|
|
$qcode="";
|
|
|
|
$comment=$this->db->get_value('select ag_title from action_gestion where ag_id=$1',
|
|
array($p_ag_id));
|
|
$array['e_client']=$qcode;
|
|
$array['e_comm']=$comment;
|
|
|
|
// retrieve info from action_detail
|
|
$a_item=$this->db->get_array('select f_id,ad_text,ad_pu,ad_quant,ad_tva_id,ad_tva_amount,j_qcode
|
|
from
|
|
action_detail
|
|
left join vw_poste_qcode using(f_id)
|
|
where
|
|
ag_id=$1', array($p_ag_id));
|
|
|
|
$array['nb_item']=($this->nb>count($a_item))?$this->nb:count($a_item);
|
|
for ($i=0; $i<count($a_item); $i++)
|
|
{
|
|
$array['e_march'.$i]=$a_item[$i]['j_qcode'];
|
|
$array['e_march'.$i.'_label']=$a_item[$i]['ad_text'];
|
|
$array['e_march'.$i.'_price']=$a_item[$i]['ad_pu'];
|
|
$array['e_march'.$i.'_tva_id']=$a_item[$i]['ad_tva_id'];
|
|
$array['e_march'.$i.'_tva_amount']=$a_item[$i]['ad_tva_amount'];
|
|
$array['e_quant'.$i]=$a_item[$i]['ad_quant'];
|
|
}
|
|
return $array;
|
|
}
|
|
|
|
/**
|
|
* Retrieve the label of an accounting
|
|
* @param $p_value tmp_pcmn.pcm_val
|
|
* @return string
|
|
*/
|
|
protected function find_label($p_value)
|
|
{
|
|
$lib=$this->db->get_value('select pcm_lib from tmp_pcmn where pcm_val=$1',
|
|
array($p_value));
|
|
return $lib;
|
|
}
|
|
|
|
/**
|
|
* Let you select the repository before confirming a sale or a purchase.
|
|
* Returns an empty string if the company doesn't use stock
|
|
* @brief Let you select the repository before confirming a sale or a purchase.
|
|
* @global type $g_parameter check if company is using stock
|
|
* @param type $p_readonly
|
|
* @param type $p_repo
|
|
* @return string
|
|
*/
|
|
public function select_depot($p_readonly, $p_repo)
|
|
{
|
|
global $g_parameter;
|
|
$r=($p_readonly==false)?'<div id="repo_div_id" style="height:185px;height:10rem;">':'<div id="repo_div_id" >';
|
|
// Show the available repository
|
|
if ($g_parameter->MY_STOCK=='Y')
|
|
{
|
|
$sel=HtmlInput::select_stock($this->db, 'repo', 'W');
|
|
$sel->readOnly=$p_readonly;
|
|
if ($p_readonly==true)
|
|
$sel->selected=$p_repo;
|
|
$r.="<p class=\"decale\">"._('Dans le dépôt')." : ";
|
|
$r.=$sel->input();
|
|
$r.='</p>';
|
|
} else
|
|
{
|
|
$r.='<span class="notice">'.'Stock non utilisé'.'</span>';
|
|
}
|
|
$r.='</div>';
|
|
return $r;
|
|
}
|
|
|
|
/**
|
|
* Create a button to encode a new operation into the same ledger
|
|
* @return string
|
|
*/
|
|
function button_new_operation()
|
|
{
|
|
$url=http_build_query(array('ac'=>$_REQUEST['ac'], 'gDossier'=>$_REQUEST['gDossier'],
|
|
'p_jrn'=>$_REQUEST['p_jrn']));
|
|
$button=HtmlInput::button_anchor(_("Nouvelle opération"),
|
|
'do.php?'.$url, "", "", "smallbutton");
|
|
return '<p>'.$button.'</p>';
|
|
}
|
|
|
|
/**
|
|
* @brief Show a button to create an operation identical to the recorded
|
|
* one. It is a form POST since it is a limit with get
|
|
*/
|
|
public function button_copy_operation()
|
|
{
|
|
echo '<FORM METHOD="POST">';
|
|
echo HtmlInput::post_to_hidden(
|
|
array("gDossier", "ac", "p_jrn", "e_client", "nb_item", "desc", "e_comm")
|
|
);
|
|
echo HtmlInput::hidden("correct", "copy");
|
|
// e_march
|
|
$http=new HttpInput();
|
|
$nb=$http->post("nb_item", "number", 0);
|
|
for ($i=0; $i<$nb; $i++)
|
|
{
|
|
echo HtmlInput::post_to_hidden(
|
|
array(
|
|
"e_march".$i,
|
|
"e_march".$i."_price",
|
|
"e_march".$i."_quant",
|
|
"e_march".$i."_label",
|
|
"e_march".$i."_tva_id",
|
|
"e_march".$i."_tva_amount",
|
|
"e_quant".$i,
|
|
"poste".$i,
|
|
"ld".$i,
|
|
"qc_".$i,
|
|
"amount".$i,
|
|
"ck".$i
|
|
));
|
|
}
|
|
echo HtmlInput::submit("copy_operation", _("Opération identique"));
|
|
|
|
echo '</FORM>';
|
|
}
|
|
|
|
/**
|
|
* Return a button to create new card, depending of the ledger
|
|
* @param $p_filter string : filter for adding : deb, cred or -1 for filter depending of the ledger
|
|
* @param $p_id_update string
|
|
*/
|
|
function add_card($p_filter, $p_id_update)
|
|
{
|
|
$js_script="this.filter='{$p_filter}';this.elementId='{$p_id_update}';this.jrn=\$('p_jrn').value; select_card_type(this);";
|
|
$str_add_button=Icon_Action::icon_add(uniqid(), $js_script);
|
|
return $str_add_button;
|
|
}
|
|
/**
|
|
* Check if a ledger is enabled , 1 for yes and 0 if disabled
|
|
*/
|
|
function is_enable()
|
|
{
|
|
return $this->db->get_value("select jrn_enable from jrn_def where jrn_def_id=$1",[$this->id]);
|
|
}
|
|
/**
|
|
* Check if the operation is used in the table quant*
|
|
* @param integer $p_grpt_id
|
|
* @param string $p_jrn_type ledger's type ACH, VEN,ODS or FIN
|
|
* @return boolean TRUE if existing info in quant*
|
|
* @Exceptions code 1000 if unknown ledger's type
|
|
*/
|
|
function use_quant_table($p_grpt_id,$p_jrn_type)
|
|
{
|
|
if ( $p_jrn_type == 'ACH')
|
|
{
|
|
$sql="select count(*) from jrnx join quant_purchase using (j_id) where j_grpt=$1";
|
|
}elseif ($p_jrn_type=='VEN')
|
|
{
|
|
$sql="select count(*) from jrnx join quant_sold using (j_id) where j_grpt=$1";
|
|
}elseif ($p_jrn_type=='FIN')
|
|
{
|
|
$sql="select count(*) from jrn join quant_fin using (jr_id) where jr_grpt_id=$1";
|
|
|
|
}elseif ($p_jrn_type=='ODS') return 0;
|
|
else
|
|
{
|
|
throw new Exception(_('Journal incorrect'),1000);
|
|
}
|
|
|
|
$count=$this->db->get_value($sql,[$p_grpt_id]);
|
|
|
|
if ($count > 0) return TRUE;
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
}
|
|
|
|
?>
|