Billing changes

This commit is contained in:
Frank Schubert
2024-08-05 21:09:13 +02:00
parent 75f75b6327
commit a6389f014f
10 changed files with 357 additions and 149 deletions

View File

@@ -48,6 +48,14 @@ $pagination_entity_name = "Billingrecords";
<label class="form-label" for="filter_product">Produkt</label>
<input type="text" class="form-control" name="filter[product]" id="filter_product" value="<?=(array_key_exists("product", $filter)) ? $filter['product'] : ""?>"/>
</div>
<div class="col-1">
<label class="form-label" for="filter_start_date_from">Periode von</label>
<input type="text" class="form-control" name="filter[start_date_from]" id="filter_start_date_from" value="<?=(array_key_exists("start_date_from", $filter)) ? $filter['start_date_from'] : ""?>"/>
</div>
<div class="col-1">
<label class="form-label" for="filter_start_date_to">Periode bis</label>
<input type="text" class="form-control" name="filter[start_date_to]" id="filter_start_date_to" value="<?=(array_key_exists("start_date_to", $filter)) ? $filter['start_date_to'] : ""?>"/>
</div>
<div class="col-2">
<label class="form-label" for="filter_show_credit">Gutschriften</label>
<select class="form-control" name="filter[show_credit]" id="filter_show_credit">
@@ -105,7 +113,7 @@ $pagination_entity_name = "Billingrecords";
</div>
<div class="float-right">
<a class="btn btn-outline-primary mb-2" href="<?=self::getUrl("Billing", "importContracts")?>">
<i class="fas fa-fw fa-file-import"></i> Aktuell verrechenbare Contracts neu importieren
<i class="fas fa-fw fa-file-import"></i> Rechnungsdatensätze aus Contracts erstellen
</a>
</div>
</div>
@@ -218,5 +226,22 @@ $pagination_entity_name = "Billingrecords";
</div>
<script>
$("#filter_start_date_from").datepicker({
orientation: "bottom",
language: 'de',
format: "dd.mm.yyyy",
showWeekDays: true,
todayBtn: 'linked',
autoclose: true
});
$("#filter_start_date_to").datepicker({
orientation: "bottom",
language: 'de',
format: "dd.mm.yyyy",
showWeekDays: true,
todayBtn: 'linked',
autoclose: true
});
</script>
<?php include(realpath(dirname(__FILE__) . "/../../$mfLayoutPackage") . "/footer.php"); ?>

View File

@@ -103,7 +103,8 @@ $pagination_entity_name = "Rechnungen";
<div class="col-12">
<h4 class="header-title">Rechnungen</h4>
<div class="float-right">
<a href="<?=self::getUrl("Invoice", "exportBmd")?>" class="btn btn-outline-primary"><i class="far fa-fw fa-file-export"></i> Rechnungsexport für BMD</a>
<a href="<?=self::getUrl("Invoice", "exportBmd", ["type" => "credit"])?>" class="btn btn-outline-danger"><i class="far fa-fw fa-file-export"></i> Provisionsexport für BMD</a>
<a href="<?=self::getUrl("Invoice", "exportBmd")?>" class="btn btn-outline-primary ml-2"><i class="far fa-fw fa-file-export"></i> Rechnungsexport für BMD</a>
</div>
</div>

View File

@@ -88,6 +88,26 @@ class BillingController extends mfBaseController {
$new_filter["invoice_id"] = null;
}
if(array_key_exists("start_date_from", $filter)) {
if($filter["start_date_from"]) {
try {
$from = DateTime::createFromFormat("d.m.Y", $filter["start_date_from"]);
} catch (Exception $e) {}
$new_filter["start_date>="] = $from->format("Y-m-d");
}
unset($filter["start_date_from"]);
}
if(array_key_exists("start_date_to", $filter)) {
if($filter["start_date_to"]) {
try {
$to = DateTime::createFromFormat("d.m.Y", $filter["start_date_to"]);
} catch (Exception $e) {}
$new_filter["start_date<="] = $to->format("Y-m-d");
}
unset($filter["start_date_to"]);
}
if(array_key_exists("customer", $filter)) {
if(array_key_exists("customer", $filter) && $filter["customer"]) {
$kunde = $this->db()->escape($filter['customer']);
@@ -135,12 +155,13 @@ class BillingController extends mfBaseController {
$now_month = date("m");
$now_day = date("d");
//$now_year = 2024;
//$now_month = 6;
//$now_day = 3; // XXX for debugging only, must be removed
$now_year = 2024;
$now_month = 7;
$now_day = 10; // XXX for debugging only, must be removed
// XXX only for 1st Billing after IVT Import
$yearly_not_before = new DateTime("$now_year-$now_month-01");
// Locking to July 2024 and keeping it for now
$yearly_not_before = new DateTime("2024-07-01");
$del = 0;
// first delete all non-invoiced billing records
@@ -303,15 +324,15 @@ class BillingController extends mfBaseController {
$end_date = (array_key_exists("end_date", $bill_data)) ? $bill_data["end_date"] : null;
$price_setup = $bill_data["price_setup"];
if($contract->billing_period > 1 && $start_date < $yearly_not_before) {
if ($contract->billing_period > 1 && $start_date < $yearly_not_before) {
// XXX only for 1st Billing after IVT Import
$this->log->debug(__METHOD__.": Ignoring Contract ".$contract->id." with start_date ".$start_date->format("Y-m-d")." because is yearly and before ".$yearly_not_before->format("Y-m-d"));
$this->log->debug(__METHOD__ . ": Ignoring Contract " . $contract->id . " with start_date " . $start_date->format("Y-m-d") . " because is yearly and before " . $yearly_not_before->format("Y-m-d"));
continue;
}
if($contract->billing_period == 1 && ($start_date->format("Y") < 2024 || ($start_date->format("Y") == 2024 && $start_date->format("m") < 7))) {
if ($contract->billing_period == 1 && ($start_date->format("Y") < 2024 || ($start_date->format("Y") == 2024 && $start_date->format("m") < 7))) {
// XXX only for 1st Billing after IVT Import
$this->log->debug(__METHOD__.": Ignoring Contract ".$contract->id." with start_date ".$start_date->format("Y-m-d")." because is before this month");
$this->log->debug(__METHOD__ . ": Ignoring Contract " . $contract->id . " with start_date " . $start_date->format("Y-m-d") . " because is before this month");
continue;
}
@@ -319,7 +340,7 @@ class BillingController extends mfBaseController {
// use cancel date as end_date
if ($cancel_date) {
$end_date = clone $cancel_date;
} elseif(!$end_date) {
} elseif (!$end_date) {
// else calculate last of month
$end_date = clone $start_date;
$end_date->modify("first day of this month");
@@ -346,7 +367,7 @@ class BillingController extends mfBaseController {
$total_days = $last_of_period->diff($first_of_period)->format("%a") + 1;
$period_days = ($end_date->diff($start_date)->format("%a")) + 1;
if($period_days < 0) continue; // don't bill for negative time range
if ($period_days < 0) continue; // don't bill for negative time range
$pc = $period_days / $total_days * 100;
$price = round($contract->price / 100 * $pc, 4);
@@ -377,7 +398,11 @@ class BillingController extends mfBaseController {
$owner = $contract->owner;
if ($contract->billingaddress_id) {
$billingaddress = $contract->billingaddress;
} else {
$billingaddress = $owner;
}
$billing_type = "invoice";
$billing_delivery = "paper";
@@ -396,10 +421,17 @@ class BillingController extends mfBaseController {
$billing_delivery = $billingaddress->billing_delivery;
}
$fibu_account_num = $contract->billingaddress->fibu_account_number;
if($contract->vatgroup_id == TT_VATGROUP_CREDIT) {
$fibu_account_num = $billingaddress->fibu_supplier_number;
if(!$fibu_account_num) {
die("Partner " . $billingaddress->customer_number . " hat keine Lieferantennummer (" . $billingaddress->getCompanyOrName().")");
}
} else {
$fibu_account_num = $billingaddress->fibu_account_number;
if(!$fibu_account_num) {
die("Keine Fibu Account Nummer in Rechnungskontakt in Contract ID ".$contract->id);
}
}
$data = [];
$data["contract_id"] = $contract->id;
@@ -430,13 +462,12 @@ class BillingController extends mfBaseController {
$data["price"] = $price;
$data["price_setup"] = $price_setup;
$data["billing_period"] = $contract->billing_period;
$data["matchcode"] = $contract->matchcode;
if(!$contract->billingaddress->country_id) {
if(!$billingaddress->country_id) {
$billcountry = CountryModel::getFirst(["isocode" => TT_HOMECOUNTRY_ISOCODE]);
} else {
$billcountry = $contract->billingaddress->country;
$billcountry = $billingaddress->country;
}
$vatgroup = $contract->vatgroup;
@@ -453,14 +484,14 @@ class BillingController extends mfBaseController {
$data["vatarea"] = $vatarea;
if($billing_type == "sepa") {
if($contract->billingaddress->sepa_date) {
$sepa_date = new DateTime("@".$contract->billingaddress->sepa_date);
if($billingaddress->sepa_date) {
$sepa_date = new DateTime("@".$billingaddress->sepa_date);
$sepa_date->setTimezone(new DateTimeZone("Europe/Vienna"));
$data["sepa_date"] = $sepa_date->format("Y-m-d");
}
if($contract->billingaddress->last_invoice_date) {
$sepa_last_date = new DateTime("@".$contract->billingaddress->last_invoice_date);
if($billingaddress->last_invoice_date) {
$sepa_last_date = new DateTime("@".$billingaddress->last_invoice_date);
$sepa_last_date->setTimezone(new DateTimeZone("Europe/Vienna"));
$data["sepa_last_date"] = $sepa_last_date->format("Y-m-d");
}
@@ -482,6 +513,7 @@ class BillingController extends mfBaseController {
/*
* Create Voice Billing, if contract has voicenumbers
*/
$voicenumbers = VoicenumberModel::search(["contract_id" => $contract->id]);
if(count($voicenumbers)) {
@@ -522,21 +554,58 @@ class BillingController extends mfBaseController {
foreach ($voicenumbers as $voicenumber) {
$vbill = BillingVoicenumberModel::getFirst(["contract_id" => $contract->id, "voicenumber" => $voicenumber->number, "start_date" => $voice_start_date->format("Y-m-d")]);
if ($vbill) {
//var_dump($vbill);exit;
continue; // number was already billed in this period
}
$calls = VoiceCallHistoryModel::getVoiceCallHistoryAsEntity(["contract_id" => $contract->id, "voice_account" => $voicenumber->number, "start" => ["from" => $voice_start_date->getTimestamp(), "to" => $voice_end_date->getTimestamp()]]);
// check for unbilled call records in earlier months
$call_dates = [];
$earlier_voice_start_date = clone $voice_start_date;
//$historic_call_count = 0;
do {
$earlier_voice_start_date->modify("-1 month");
// not before first billing in thetool
if($earlier_voice_start_date->format("Y-m-d") < "2024-07-01") break;
$earlier_voice_end_date = clone $earlier_voice_start_date;
$earlier_voice_end_date->modify("last day of this month");
$earlier_voice_end_date->setTime(23, 59, 59);
$historic_call_count = VoiceCallHistoryModel::countVoiceCallHistory(["contract_id" => $contract->id, "voice_account" => $voicenumber->number, "start" => ["from" => $earlier_voice_start_date->getTimestamp(), "to" => $earlier_voice_end_date->getTimestamp()]]);
//var_dump($historic_call_count);exit;
if ($historic_call_count) {
$call_dates[] = [
"start_date" => clone $earlier_voice_start_date,
"end_date" => clone $earlier_voice_end_date,
];
}
//var_dump($historic_call_count > 0);exit;
} while ($historic_call_count > 0);
$call_dates = array_reverse($call_dates);
$call_dates[] = [
"start_date" => clone $voice_start_date,
"end_date" => clone $voice_end_date,
];
if(count($call_dates) > 1) {
var_dump($call_dates);exit;
}
foreach ($call_dates as $call_date) {
$calls = VoiceCallHistoryModel::getVoiceCallHistoryAsEntity(["contract_id" => $contract->id, "voice_account" => $voicenumber->number, "start" => ["from" => $call_date["start_date"]->getTimestamp(), "to" => $call_date["end_date"]->getTimestamp()]]);
foreach ($calls as $call) {
//var_dump($call);
$number = $call->voice_account;
$dest_nummer = $call->destination;
if(substr($dest_nummer, 0, 2) == "00") {
if (substr($dest_nummer, 0, 2) == "00") {
$dest_nummer = substr($dest_nummer, 2);
}
if(substr($dest_nummer, 0, 1) == "+") {
if (substr($dest_nummer, 0, 1) == "+") {
$dest_nummer = substr($dest_nummer, 1);
}
@@ -544,8 +613,8 @@ class BillingController extends mfBaseController {
$destination = $destinations_cache[$dest_nummer];
} else {
$destination = $voiceplan->getDestinationByNumber($dest_nummer);
if(!$destination) {
die("Destination für Zielrufnummer ".$call->destination." nicht gefunden");
if (!$destination) {
die("Destination für Zielrufnummer " . $call->destination . " nicht gefunden");
}
$destinations_cache[$dest_nummer] = $destination;
}
@@ -553,8 +622,8 @@ class BillingController extends mfBaseController {
$zone = $destination->voiceplanzone;
if(!$zone) {
die("Keine Zone für Destination ".$dest_nummer." gefunden");
if (!$zone) {
die("Keine Zone für Destination " . $dest_nummer . " gefunden");
}
//var_dump($zone);
@@ -565,7 +634,7 @@ class BillingController extends mfBaseController {
$inc = $zone->increment;
$billable_duration = $call->duration;
if($billable_duration <= 0) continue;
if ($billable_duration <= 0) continue;
// calculate price of first duration unit
// then subtract first minimum duration from duration
@@ -574,7 +643,7 @@ class BillingController extends mfBaseController {
$billable_duration -= $inc_first;
// calculate price of remaining duration and make sure to bill in full duration units
if($billable_duration > 0) {
if ($billable_duration > 0) {
$multi = ceil($billable_duration / $inc);
$call_price += ($multi * $inc) * $sec_price;
}
@@ -591,7 +660,9 @@ class BillingController extends mfBaseController {
"zone_total" => 0,
"increment_first" => $zone->increment_first,
"increment" => $zone->increment,
"count" => 0
"count" => 0,
"start_date" => $call_date["start_date"]->format("Y-m-d"),
"end_date" => $call_date["end_date"]->format("Y-m-d"),
];
}
@@ -600,6 +671,7 @@ class BillingController extends mfBaseController {
$voicebills[$number][$zone->id]["duration"] += $call->duration;
}
}
}
// save to BillingVoicenumber
foreach($voicebills as $vbnumber => $zones) {
@@ -609,8 +681,8 @@ class BillingController extends mfBaseController {
$vbdata["billing_id"] = $billing->id;
$vbdata["contract_id"] = $contract->id;
$vbdata["voicenumber"] = $vbnumber;
$vbdata["start_date"] = $voice_start_date->format("Y-m-d");
$vbdata["end_date"] = $voice_end_date->format("Y-m-d");
$vbdata["start_date"] = $vb["start_date"];
$vbdata["end_date"] = $vb["end_date"];
$vbdata["voiceplan"] = $vb["voiceplan"];
$vbdata["zone"] = $vb["zone_name"];
$vbdata["call_count"] = $vb["count"];
@@ -633,14 +705,6 @@ class BillingController extends mfBaseController {
}
/*foreach(VoiceCallHistoryModel::getVoiceCallHistoryAsEntity(["contract_id" => $contract->id, "start" => ["from" => $start_date->getTimestamp()]]) as $call) {
// find BillingVoicenumber record for this call
$vbill = BillingVoicenumberModel::getFirst(["contract_id" => $contract->id, "voicenumber" => $call->voice_account, "start_date" => ]);
if($vbill) {
}
}*/
}

View File

@@ -205,10 +205,10 @@ class BillingModel {
$items = [];
$where = self::getSqlFilter($filter);
$sql = "SELECT owner_id, billingaddress_id, billing_type, billing_delivery FROM Billing
$sql = "SELECT owner_id, billingaddress_id, billing_type, billing_delivery, vatgroup_id FROM Billing
WHERE $where
GROUP BY owner_id, billingaddress_id, billing_type, billing_delivery
ORDER BY owner_id, billingaddress_id, billing_type, billing_delivery";
GROUP BY owner_id, billingaddress_id, billing_type, billing_delivery, vatgroup_id
ORDER BY owner_id, billingaddress_id, billing_type, billing_delivery, vatgroup_id";
//var_dump($sql);exit;
$res = $db->query($sql);
if($db->num_rows($res)) {
@@ -217,7 +217,8 @@ class BillingModel {
"owner_id" => $data->owner_id,
"billingaddress_id" => $data->billingaddress_id,
"billing_type" => $data->billing_type,
"billing_delivery" => $data->billing_delivery
"billing_delivery" => $data->billing_delivery,
"vatgroup_id" => $data->vatgroup_id
];
}
}
@@ -490,11 +491,6 @@ class BillingModel {
}
}
if(array_key_exists("add-where", $filter)) {
$where .= " ".$filter['add-where'];
}
if(array_key_exists("billing_type", $filter)) {
$billing_type = $filter['billing_type'];
if(is_numeric($billing_type)) {
@@ -516,6 +512,25 @@ class BillingModel {
}
}
if(array_key_exists("vatgroup_id", $filter)) {
$vatgroup_id = $filter['vatgroup_id'];
if(is_numeric($vatgroup_id)) {
$where .= " AND Billing.vatgroup_id = $vatgroup_id";
}
}
if(array_key_exists("!vatgroup_id", $filter)) {
$vatgroup_id = $filter['!vatgroup_id'];
if(is_numeric($vatgroup_id)) {
$where .= " AND Billing.vatgroup_id <> $vatgroup_id";
}
}
if(array_key_exists("add-where", $filter)) {
$where .= " ".$filter['add-where'];
}
//var_dump($filter, $where);exit;
return $where;
}

View File

@@ -768,7 +768,7 @@ class ContractController extends mfBaseController
$contract_data = [];
$contract_data["owner_id"] = (int)$r->owner_id;
$contract_data["billingaddress_id"] = ($r->billingaddress_id) ? (int)$r->billingaddress_id : null;
$contract_data["billingaddress_id"] = ($r->billingaddress_id) ? (int)$r->billingaddress_id : $r->owner_id;
$contract_data["product_id"] = (int)$r->product_id;
$contract_data["matchcode"] = $r->matchcode;
$contract_data["product_name"] = $r->product_name;

View File

@@ -189,7 +189,7 @@ class ContractModel {
}
/**
* To create Credit Contract for Partner
* To create Credit Contract for Partner (NNE)
*
* @param Contract $contract
* @return bool|Contract

View File

@@ -163,28 +163,39 @@ class InvoiceController extends mfBaseController {
$i = 0;
$p = 0;
$base_rows_bill = BillingModel::getInvoiceBaseData(['invoice_id' => null, "!vatgroup_id" => TT_VATGROUP_CREDIT]);
$base_rows_credit = BillingModel::getInvoiceBaseData(['invoice_id' => null, "vatgroup_id" => TT_VATGROUP_CREDIT]);
$base_rows = array_merge($base_rows_bill, $base_rows_credit);
// get pairs of owner_id and billingaddress_id, so each will be its own invoice
foreach (BillingModel::getInvoiceBaseData(['invoice_id' => null]) as $base) {
//var_dump($base); continue;
foreach ($base_rows as $base) {
//var_dump($base);exit;
$owner_id = $base["owner_id"];
$billingaddress_id = $base["billingaddress_id"];
$billing_type = $base["billing_type"];
$billing_delivery = $base["billing_delivery"];
$base_vatgroup_id = $base["vatgroup_id"];
$bill_positions = [];
$credit_positions = [];
$invoice_voicenumbers = [];
$billing_rows = BillingModel::search(["owner_id" => $owner_id,
$bill_filter = ["owner_id" => $owner_id,
"billingaddress_id" => $billingaddress_id,
"billing_type" => $billing_type,
"billing_delivery" => $billing_delivery,
"invoice_id" => null]);
"invoice_id" => null];
if (!count($billing_rows)) {
die("Keine nicht verrechneten Billing records für billingaddress_id");
if($base_vatgroup_id == TT_VATGROUP_CREDIT) {
$bill_filter["vatgroup_id"] = TT_VATGROUP_CREDIT;
} else {
$bill_filter["!vatgroup_id"] = TT_VATGROUP_CREDIT;
}
$billing_rows = BillingModel::search($bill_filter);
if(!count($billing_rows)) continue;
//var_dump($billing_rows);exit;
//var_dump($owner_id, $billingaddress_id, $bills);exit;
$invoice_data = [];
@@ -192,6 +203,7 @@ class InvoiceController extends mfBaseController {
$price_total_sum = 0;
foreach($billing_rows as $bill) {
$vatgroup_id = $bill->vatgroup_id;
$vatarea = $bill->vatarea;
$invoice_vatrate = $bill->vatgroup->rates[$vatarea]->rate;
@@ -263,7 +275,7 @@ class InvoiceController extends mfBaseController {
$voicebills = BillingVoicenumberModel::search(["billing_id" => $bill->id]);
//var_dump($voicebills);exit;
if(count($voicebills)) {
if (count($voicebills)) {
$voice_start_date = reset($voicebills)->start_date;
$voice_end_date = reset($voicebills)->end_date;
$voiceplan = reset($voicebills)->voiceplan;
@@ -271,7 +283,7 @@ class InvoiceController extends mfBaseController {
$inc_first = reset($voicebills)->increment_first;
$voice_rows = [];
foreach($voicebills as $voicebill) {
foreach ($voicebills as $voicebill) {
$number = $voicebill->voicenumber;
$zone = $voicebill->zone;
$call_count = $voicebill->call_count;
@@ -280,15 +292,15 @@ class InvoiceController extends mfBaseController {
$price_total = $voicebill->price_total;
if(!array_key_exists($number, $voice_rows)) {
if (!array_key_exists($number, $voice_rows)) {
$voice_rows[$number] = [];
}
if(!array_key_exists($zone, $voice_rows[$number])) {
if (!array_key_exists($zone, $voice_rows[$number])) {
$voice_rows[$number][$zone] = [];
}
if(!array_key_exists($price, $voice_rows[$number][$zone])) {
if (!array_key_exists($price, $voice_rows[$number][$zone])) {
$voice_rows[$number][$zone][$price] = [];
$voice_rows[$number][$zone][$price]["call_count"] = 0;
$voice_rows[$number][$zone][$price]["duration"] = 0;
@@ -304,7 +316,7 @@ class InvoiceController extends mfBaseController {
$total_voice_price = 0;
foreach($voice_rows as $number => $zones) {
foreach ($voice_rows as $number => $zones) {
$voicenumber_data = [];
$voicenumber_data["voicenumber"] = $number;
$voicenumber_data["start_date"] = $voice_start_date;
@@ -317,9 +329,8 @@ class InvoiceController extends mfBaseController {
$invoice_voicenumber->voicenumberzones = [];
foreach($zones as $zone => $prices) {
foreach($prices as $price => $row_values) {
foreach ($zones as $zone => $prices) {
foreach ($prices as $price => $row_values) {
$zone_data = [];
$zone_data["zone"] = $zone;
$zone_data["call_count"] = $row_values["call_count"];
@@ -327,7 +338,7 @@ class InvoiceController extends mfBaseController {
$zone_data["price"] = $price;
$zone_data["price_total"] = $row_values["price_total"];
$zone_data["price_total_gross"] = $row_values["price_total"];
if($invoice_vatrate) {
if ($invoice_vatrate) {
$zone_data["price_total_gross"] = $row_values["price_total"] + (($row_values["price_total"] / 100) * $invoice_vatrate);
}
$zone_data["vatrate"] = $invoice_vatrate;
@@ -374,6 +385,8 @@ class InvoiceController extends mfBaseController {
$invoice_data["billingaddress_id"] = $billingaddress_id;
$invoice_data["customer_number"] = $bill->customer_number;
$invoice_data["fibu_account_number"] = $bill->fibu_account_number;
$invoice_data["sepa_date"] = $bill->sepa_date;
$invoice_data["sepa_id"] = $bill->sepa_id;
$invoice_data["sepa_last_date"] = $bill->sepa_last_date;
@@ -395,6 +408,7 @@ class InvoiceController extends mfBaseController {
$invoice_data["bank_account_bic"] = $bill->bank_account_bic;
$invoice_data["total"] = 0;
$invoice_data["total_gross"] = 0;
$invoice_data["vatgroup_id"] = $vatgroup_id;
//$invoice_data["total_vat"] = 0;
@@ -404,7 +418,6 @@ class InvoiceController extends mfBaseController {
$invoice_data["fibu_taxcode"] = $bill->vatgroup->rates[$vatarea]->taxcode;
$invoice_data["tax_text"] = $bill->vatgroup->rates[$vatarea]->invoice_text;
}
@@ -414,7 +427,7 @@ class InvoiceController extends mfBaseController {
continue;
}*/
//var_dump($invoice_data);exit;
/*
* *******************************
* Save invoice and add positions
@@ -426,6 +439,7 @@ class InvoiceController extends mfBaseController {
try {
if (!$invoice->save()) {
var_dump($invoice);
$invoice->rollbackTransaction();
die("Error saving Invoice");
}
@@ -556,14 +570,37 @@ class InvoiceController extends mfBaseController {
* 0;234325;40010;TEST-2;23.01.2024;GU;1;20;1;-1,2;0,2;GSTEST-01;;;;;;;;10;0
*/
$csv_header = "satzart;konto;gkonto;belegnr;belegdatum;buchsymbol;buchcode;prozent;steuercode;betrag;steuer;text;kost;bank-iban-nr;bank-swiftcode;bank-mandatsid;bank-mandatsdatum;bank-mandatskz;bank-letztereinzug;zvsperre;bankeinzug";
//var_dump($this->request->type);exit;
$type = "normal";
if($this->request->type == "credit") {
$type = "credit";
}
$csv_header = "satzart;konto;gkonto;belegnr;belegdatum;buchsymbol;buchcode;prozent;steuercode;betrag;steuer;text;kost;";
if($type == "credit") {
$csv_header .= "extbelegnr";
} else {
$csv_header .= "bank-iban-nr;bank-swiftcode;bank-mandatsid;bank-mandatsdatum;bank-mandatskz;bank-letztereinzug;zvsperre;bankeinzug";
}
$csv_out = "";
if(!InvoiceModel::search(["bmd_export_date" => null])) {
$filter = [
"bmd_export_date" => null,
];
if($type == "credit") {
$filter["vatgroup_id"] = TT_VATGROUP_CREDIT;
} else {
$filter["!vatgroup_id"] = TT_VATGROUP_CREDIT;
}
if(!InvoiceModel::search($filter)) {
$this->layout()->setFlash("Keine neuen Rechnungen seit letztem BMD-Export vorhanden.");
$this->redirect("Invoice");
}
foreach(InvoiceModel::search(["bmd_export_date" => null]) as $invoice) {
foreach(InvoiceModel::search($filter) as $invoice) {
if($invoice->bmd_export_date) {
die("wtf");
}
@@ -579,9 +616,10 @@ class InvoiceController extends mfBaseController {
$buchsymbol = "GU";
} else {
$buchsymbol = "AR";
}
$fibu_account = $invoice->fibu_account_number;
$buchungstext = "[".$invoice->customer_number."]";
if($invoice->company) {
$buchungstext .= " ".$invoice->company;
@@ -590,7 +628,7 @@ class InvoiceController extends mfBaseController {
}
$buchungstext = str_replace(["\n","\r", ";"], "", $buchungstext);
$buchcode = "1";
$is_sepa = ($invoice->billing_type == "sepa");
$iban = "";
@@ -620,19 +658,32 @@ class InvoiceController extends mfBaseController {
}
if($type == "credit") {
$buchsymbol = "ER";
$buchcode = "2";
if($invoice->fibu_supplier_number) {
$this->layout()->setFlash("Lieferant ".$invoice->customer_number." - ".$invoice->company." ".$invoice->firstname." ".$invoice->lastname. " hat keine Lieferantennummer! Bitte nachtragen.");
$this->redirect("Invoice");
}
$fibu_account = $invoice->fibu_account_number;
}
$csv_out .= "0;";
$csv_out .= $invoice->fibu_account_number.";";
$csv_out .= $fibu_account.";";
$csv_out .= $invoice->fibu_cost_account.";";
$csv_out .= $invoice->invoice_number.";";
$csv_out .= date("d.m.Y", $invoice->invoice_date).";";
$csv_out .= $buchsymbol.";";
$csv_out .= "1;";
$csv_out .= $buchcode.";";
$csv_out .= $vatrate.";";
$csv_out .= $invoice->fibu_taxcode.";";
$csv_out .= number_format($invoice->total_gross, 2, ",", "").";";
$csv_out .= number_format($vat, 2, ",", "").";";
$csv_out .= $buchungstext.";";
$csv_out .= ";"; // kostenstelle
if($type != "credit") {
$csv_out .= $iban.";";
$csv_out .= $bic.";";
$csv_out .= $sepa_id.";";
@@ -641,6 +692,7 @@ class InvoiceController extends mfBaseController {
$csv_out .= ($last_invoice_date ? $last_invoice_date->format("d.m.Y") : "").";";
$csv_out .= ($is_sepa ? 0 : 10).";";
$csv_out .= ($is_sepa ? 1 : 0);
}
$csv_out .= "\n";
}
@@ -648,7 +700,12 @@ class InvoiceController extends mfBaseController {
$this->redirect("Invoice");*/
header("Content-type: text/csv");
if($type == "credit") {
header('Content-disposition: attachment; filename="tt-prov-export-bmd-'.date('Y-m-d_H-i-s').'.csv"');
} else {
header('Content-disposition: attachment; filename="tt-rech-export-bmd-'.date('Y-m-d_H-i-s').'.csv"');
}
echo $csv_header."\n".$csv_out;
exit;

View File

@@ -32,6 +32,7 @@ class InvoiceModel {
public $bank_account_bic;
public $total;
public $total_gross;
public $vatgroup_id;
public $bmd_export_date;
public $date_delivered;
public $total_vat;
@@ -407,6 +408,19 @@ class InvoiceModel {
}
}
if(array_key_exists("vatgroup_id", $filter)) {
$vatgroup_id = $filter['vatgroup_id'];
if(is_numeric($vatgroup_id)) {
$where .= " AND Invoice.vatgroup_id = '$vatgroup_id'";
}
}
if(array_key_exists("!vatgroup_id", $filter)) {
$vatgroup_id = $filter['!vatgroup_id'];
if(is_numeric($vatgroup_id)) {
$where .= " AND Invoice.vatgroup_id <> '$vatgroup_id'";
}
}
if(array_key_exists("add-where", $filter)) {
$where .= " ".$filter['add-where'];

View File

@@ -152,6 +152,7 @@ class VoiceCallHistoryModel {
public static function countVoiceCallHistory($filters) {
$db = FronkDB::singleton();
$sql = "SELECT COUNT(*) as `total_rows` FROM `VoiceCallHistory` WHERE 1 " . self::getSqlFilter($filters);
mfLoghandler::singleton()->debug($sql);
$result = $db->query($sql);
return $result->fetch_assoc()['total_rows'];
}

View File

@@ -0,0 +1,31 @@
<?php
declare(strict_types=1);
use Phinx\Migration\AbstractMigration;
final class InvoiceAddVatgroupId extends AbstractMigration
{
public function up(): void
{
if($this->getEnvironment() == "thetool") {
$table = $this->table('Invoice');
$table->addColumn("vatgroup_id", "integer", ["null" => false, "after" => "total_gross"]);
$table->update();
}
if($this->getEnvironment() == "addressdb") {
}
}
public function down(): void
{
if($this->getEnvironment() == "thetool") {
$this->table('Invoice')->removeColumn("vatgroup_id")->save();
}
if($this->getEnvironment() == "addressdb") {
}
}
}