migrate to gtea from bistbucket
This commit is contained in:
793
app/Helpers/Helper.php
Normal file
793
app/Helpers/Helper.php
Normal file
@@ -0,0 +1,793 @@
|
||||
<?php
|
||||
|
||||
use App\Models\User;
|
||||
use App\Models\Branch;
|
||||
use App\Models\Option;
|
||||
use App\Models\Business;
|
||||
use App\Models\Currency;
|
||||
use App\Models\Transaction;
|
||||
use App\Models\UserCurrency;
|
||||
use Kreait\Firebase\Factory;
|
||||
use App\Models\PlanSubscribe;
|
||||
use App\Models\ProductSetting;
|
||||
use Illuminate\Support\Facades\DB;
|
||||
use Illuminate\Support\Facades\Log;
|
||||
use Nwidart\Modules\Facades\Module;
|
||||
use Illuminate\Support\Facades\Date;
|
||||
use Illuminate\Support\Facades\Http;
|
||||
use Illuminate\Support\Facades\Cache;
|
||||
use App\Notifications\SendNotification;
|
||||
use Illuminate\Support\Facades\Artisan;
|
||||
use Modules\MarketingAddon\App\Models\Sim;
|
||||
use Kreait\Firebase\Messaging\CloudMessage;
|
||||
use Illuminate\Support\Facades\Notification;
|
||||
use Modules\MarketingAddon\App\Models\Device;
|
||||
use Modules\MarketingAddon\App\Models\Smsgateway;
|
||||
|
||||
function cache_remember(string $key, callable $callback, int $ttl = 5000): mixed
|
||||
{
|
||||
return cache()->remember($key, env('CACHE_LIFETIME', $ttl), $callback);
|
||||
}
|
||||
|
||||
function get_option($key)
|
||||
{
|
||||
return cache_remember($key, function () use ($key) {
|
||||
return Option::where('key', $key)->first()->value ?? [];
|
||||
});
|
||||
}
|
||||
|
||||
function invoice_setting()
|
||||
{
|
||||
return get_option('invoice_setting_' . auth()->user()->business_id);
|
||||
}
|
||||
|
||||
function invoice_language()
|
||||
{
|
||||
return get_option('invoice_language_' . auth()->user()->business_id);
|
||||
}
|
||||
|
||||
function product_setting()
|
||||
{
|
||||
$businessId = auth()->user()->business_id;
|
||||
$cacheKey = 'product_setting_' . $businessId;
|
||||
|
||||
return cache()->remember($cacheKey, 60, function () use ($businessId) {
|
||||
$productSetting = ProductSetting::where('business_id', $businessId)->first();
|
||||
|
||||
if ($productSetting) {
|
||||
$productSetting->modules = $productSetting->modules ?? [];
|
||||
}
|
||||
|
||||
return $productSetting;
|
||||
});
|
||||
}
|
||||
|
||||
function is_module_enabled(?array $modules, string $key): bool
|
||||
{
|
||||
// Keys that should default to true if not set
|
||||
$defaultTrueKeys = [
|
||||
'show_product_category',
|
||||
'show_product_stock',
|
||||
'show_exclusive_price',
|
||||
'show_inclusive_price',
|
||||
'show_profit_percent',
|
||||
'show_product_sale_price',
|
||||
'show_product_wholesale_price',
|
||||
'show_product_dealer_price',
|
||||
'show_action',
|
||||
];
|
||||
|
||||
if (in_array($key, $defaultTrueKeys)) {
|
||||
return !isset($modules[$key]) || (bool) $modules[$key];
|
||||
}
|
||||
|
||||
// All other keys: show only if explicitly set to true
|
||||
return isset($modules[$key]) && (bool) $modules[$key];
|
||||
}
|
||||
|
||||
function formatted_date(?string $date = null, string $format = 'd M, Y'): ?string
|
||||
{
|
||||
return !empty($date) ? Date::parse($date)->format($format) : null;
|
||||
}
|
||||
|
||||
function formatted_time(?string $time = null, string $format = 'h:i A'): ?string
|
||||
{
|
||||
return !empty($time) ? Date::parse($time)->format($format) : null;
|
||||
}
|
||||
|
||||
function sendNotification($id, $url, $message, $user = null)
|
||||
{
|
||||
$notify = [
|
||||
'id' => $id,
|
||||
'url' => $url,
|
||||
'user' => $user,
|
||||
'message' => $message,
|
||||
];
|
||||
|
||||
$notify_user = User::where('role', 'superadmin')->first();
|
||||
Notification::send($notify_user, new SendNotification($notify));
|
||||
}
|
||||
|
||||
function sendNotifyToUser($id, $url, $message, $user)
|
||||
{
|
||||
$notify = [
|
||||
'id' => $id,
|
||||
'url' => $url,
|
||||
'user' => $user,
|
||||
'message' => $message,
|
||||
];
|
||||
|
||||
$notify_user = User::where('business_id', $user)->first();
|
||||
Notification::send($notify_user, new SendNotification($notify));
|
||||
}
|
||||
|
||||
function currency_format($amount, $type = "icon", $decimals = 2, $currency = null, $abbreviate = false, $apply_rounding = false)
|
||||
{
|
||||
$currency = $currency ?? default_currency();
|
||||
|
||||
if ($apply_rounding) {
|
||||
$amount = sale_rounding((float) $amount);
|
||||
}
|
||||
|
||||
if ($abbreviate) {
|
||||
$amount = format_number($amount, $decimals);
|
||||
} else {
|
||||
$has_fraction = $amount != floor($amount);
|
||||
$amount = $has_fraction ? number_format($amount, $decimals) : number_format($amount, 0);
|
||||
}
|
||||
|
||||
if ($type == "icon" || $type == "symbol") {
|
||||
if ($currency->position == "right") {
|
||||
return $amount . $currency->symbol;
|
||||
} else {
|
||||
return $currency->symbol . $amount;
|
||||
}
|
||||
} else {
|
||||
if ($currency->position == "right") {
|
||||
return $amount . ' ' . $currency->code;
|
||||
} else {
|
||||
return $currency->code . ' ' . $amount;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
function format_number(float|int $number, int $decimals = 2): string
|
||||
{
|
||||
if ($number >= 1e9) {
|
||||
return remove_trailing_zeros($number / 1e9, $decimals) . "B";
|
||||
} elseif ($number >= 1e6) {
|
||||
return remove_trailing_zeros($number / 1e6, $decimals) . "M";
|
||||
} elseif ($number >= 1e3) {
|
||||
return remove_trailing_zeros($number / 1e3, $decimals) . "K";
|
||||
} else {
|
||||
return remove_trailing_zeros($number, $decimals);
|
||||
}
|
||||
}
|
||||
|
||||
function remove_trailing_zeros(float|int $number, int $decimals = 2): string
|
||||
{
|
||||
return rtrim(rtrim(number_format($number, $decimals, '.', ''), '0'), '.');
|
||||
}
|
||||
|
||||
function amountInWords(float $amount, int $decimals = 2): string
|
||||
{
|
||||
if (!extension_loaded('intl')) {
|
||||
return '';
|
||||
}
|
||||
|
||||
$has_fraction = fmod($amount, 1) != 0;
|
||||
$amount = $has_fraction ? round($amount, $decimals) : round($amount);
|
||||
|
||||
$formatter = new \NumberFormatter('en_US', \NumberFormatter::SPELLOUT);
|
||||
$words = $formatter->format($amount);
|
||||
|
||||
return $words . ' ' . (business_currency()->name ?? '');
|
||||
}
|
||||
|
||||
function convert_money($amount, $currency)
|
||||
{
|
||||
if ($currency->code == default_currency('code') || $amount == 0) {
|
||||
return round($amount, 2);
|
||||
} else {
|
||||
return $amount * $currency->rate / default_currency()->rate;
|
||||
}
|
||||
}
|
||||
|
||||
function default_currency($key = null, ?Currency $currency = null): object|int|string
|
||||
{
|
||||
$currency = $currency ?? cache_remember('default_currency', function () {
|
||||
$currency = Currency::whereIsDefault(1)->first();
|
||||
|
||||
if (!$currency) {
|
||||
$currency = (object) ['name' => 'US Dollar', 'code' => 'USD', 'rate' => 1, 'symbol' => '$', 'position' => 'left', 'status' => true, 'is_default' => true,];
|
||||
}
|
||||
|
||||
return $currency;
|
||||
});
|
||||
|
||||
return $key ? $currency->$key : $currency;
|
||||
}
|
||||
|
||||
function paymentReminderMessage($sale, $business, $customer_name = 'Customer')
|
||||
{
|
||||
$message = get_option('sms-templates-' . $business->id)['purchase_sms'] ?? "Hello [customer_name], kindly clear your outstanding balance of [amount]. For details, contact us at [business_phone].";
|
||||
|
||||
$data = [
|
||||
'customer_name' => $customer_name,
|
||||
'invoice_no' => $sale->invoiceNumber,
|
||||
'due_amount' => $sale->totalAmount,
|
||||
'date' => formatted_date($sale->saleDate),
|
||||
'business_name' => $business->companyName,
|
||||
'business_phone' => $business->phoneNumber,
|
||||
];
|
||||
|
||||
foreach ($data as $key => $value) {
|
||||
$message = str_replace("[$key]", $value, $message);
|
||||
}
|
||||
|
||||
return $message;
|
||||
}
|
||||
|
||||
function sendMessage($numbers, $contents, $business_id = null)
|
||||
{
|
||||
$gateways = Smsgateway::where('business_id', $business_id ?? auth()->user()->business_id)->where('status', 1)->get();
|
||||
|
||||
foreach ($gateways as $gateway) {
|
||||
$gateway->namespace::send_message($gateway, $numbers, $contents);
|
||||
session()->put('gateway_id', $gateway->id);
|
||||
|
||||
return [
|
||||
'status' => true,
|
||||
'message' => "Message send successfully."
|
||||
];
|
||||
}
|
||||
}
|
||||
|
||||
function sendPushNotify($request, int|string $total_numbers, ?int $device_id = null): bool
|
||||
{
|
||||
$credentialPath = public_path('uploads/service-account-credentials.json');
|
||||
if (!file_exists($credentialPath))
|
||||
return false;
|
||||
|
||||
$firebase = (new Factory)->withServiceAccount($credentialPath)->createMessaging();
|
||||
$tokens = [];
|
||||
|
||||
if ($device_id) {
|
||||
$tokens = Device::whereNotNull('device_token')
|
||||
->where('business_id', auth()->user()->business_id)
|
||||
->where('id', $device_id)
|
||||
->pluck('device_token');
|
||||
} else {
|
||||
$device_ids = Sim::whereIn('id', $request->sim_ids)->pluck('device_id')->unique();
|
||||
$tokens = Device::whereNotNull('device_token')->whereIn('id', $device_ids)->pluck('device_token');
|
||||
}
|
||||
|
||||
foreach ($tokens as $token) {
|
||||
$message = CloudMessage::fromArray([
|
||||
'notification' => [
|
||||
'title' => 'New ' . $request->type . ' has been created.',
|
||||
'body' => $total_numbers . ' ' . $request->type . ' has been created.',
|
||||
],
|
||||
'data' => ['type' => $request->type],
|
||||
'token' => $token,
|
||||
]);
|
||||
|
||||
try {
|
||||
$firebase->send($message);
|
||||
} catch (\Kreait\Firebase\Exception\MessagingException $e) {
|
||||
Log::error('Firebase Notification Error', ['error' => $e->getMessage()]);
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
function restorePublicImages()
|
||||
{
|
||||
if (!env('DEMO_MODE')) {
|
||||
return true;
|
||||
}
|
||||
|
||||
DB::table('sales')->where('business_id', 1)->delete();
|
||||
DB::table('sale_returns')->where('business_id', 1)->delete();
|
||||
DB::table('purchases')->where('business_id', 1)->delete();
|
||||
DB::table('purchase_returns')->where('business_id', 1)->delete();
|
||||
DB::table('due_collects')->where('business_id', 1)->delete();
|
||||
DB::table('parties')->where('business_id', 1)->delete();
|
||||
DB::table('expense_categories')->where('business_id', 1)->delete();
|
||||
DB::table('income_categories')->where('business_id', 1)->delete();
|
||||
|
||||
Artisan::call('db:seed', ['--class' => 'DemoSeeder']);
|
||||
}
|
||||
|
||||
if (!function_exists('formatTimeToWords')) {
|
||||
function formatTimeToWords(string|null $time): string
|
||||
{
|
||||
if (empty($time)) {
|
||||
return '0';
|
||||
}
|
||||
|
||||
if (!preg_match('/^\d{1,2}:\d{2}(:\d{2})?$/', $time)) {
|
||||
return '0';
|
||||
}
|
||||
|
||||
$parts = explode(':', $time);
|
||||
|
||||
$hours = isset($parts[0]) ? (int) $parts[0] : 0;
|
||||
$minutes = isset($parts[1]) ? (int) $parts[1] : 0;
|
||||
|
||||
$hourString = $hours == 1 ? 'hour' : 'hours';
|
||||
$minuteString = $minutes == 1 ? 'minute' : 'minutes';
|
||||
|
||||
$formattedTime = [];
|
||||
|
||||
if ($hours > 0) {
|
||||
$formattedTime[] = "$hours $hourString";
|
||||
}
|
||||
|
||||
if ($minutes > 0) {
|
||||
$formattedTime[] = "$minutes $minuteString";
|
||||
}
|
||||
|
||||
return empty($formattedTime) ? '0' : implode(' and ', $formattedTime);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
function languages()
|
||||
{
|
||||
return [
|
||||
'en' => ['name' => 'English', 'flag' => 'us'],
|
||||
'ar' => ['name' => 'Arabic', 'flag' => 'sa'],
|
||||
'bn' => ['name' => 'Bengali', 'flag' => 'bd'],
|
||||
'zh' => ['name' => 'Chinese', 'flag' => 'cn'],
|
||||
'fr' => ['name' => 'French', 'flag' => 'fr'],
|
||||
'de' => ['name' => 'German', 'flag' => 'de'],
|
||||
'hi' => ['name' => 'Hindi', 'flag' => 'in'],
|
||||
'es' => ['name' => 'Spanish', 'flag' => 'es'],
|
||||
'ja' => ['name' => 'Japanese', 'flag' => 'jp'],
|
||||
'rum' => ['name' => 'Romanian', 'flag' => 'ro'],
|
||||
'vi' => ['name' => 'Vietnamese', 'flag' => 'vn'],
|
||||
'it' => ['name' => 'Italian', 'flag' => 'it'],
|
||||
'th' => ['name' => 'Thai', 'flag' => 'th'],
|
||||
'bs' => ['name' => 'Bosnian', 'flag' => 'ba'],
|
||||
'nl' => ['name' => 'Dutch', 'flag' => 'nl'],
|
||||
'pt' => ['name' => 'Portuguese', 'flag' => 'pt'],
|
||||
'pl' => ['name' => 'Polish', 'flag' => 'pl'],
|
||||
'he' => ['name' => 'Hebrew', 'flag' => 'il'],
|
||||
'hu' => ['name' => 'Hungarian', 'flag' => 'hu'],
|
||||
'fi' => ['name' => 'Finnish', 'flag' => 'fi'],
|
||||
'el' => ['name' => 'Greek', 'flag' => 'gr'],
|
||||
'ko' => ['name' => 'Korean', 'flag' => 'kr'],
|
||||
'ms' => ['name' => 'Malay', 'flag' => 'my'],
|
||||
'id' => ['name' => 'Indonesian', 'flag' => 'id'],
|
||||
'fa' => ['name' => 'Persian', 'flag' => 'ir'],
|
||||
'tr' => ['name' => 'Turkish', 'flag' => 'tr'],
|
||||
'sr' => ['name' => 'Serbian', 'flag' => 'rs'],
|
||||
'km' => ['name' => 'Khmer', 'flag' => 'khm'],
|
||||
'uk' => ['name' => 'Ukrainian', 'flag' => 'ua'],
|
||||
'lo' => ['name' => 'Lao', 'flag' => 'la'],
|
||||
'ru' => ['name' => 'Russian', 'flag' => 'ru'],
|
||||
'cs' => ['name' => 'Czech', 'flag' => 'cz'],
|
||||
'kn' => ['name' => 'Kannada', 'flag' => 'ka'],
|
||||
'mr' => ['name' => 'Marathi', 'flag' => 'mh'],
|
||||
'sv' => ['name' => 'Swedish', 'flag' => 'se'],
|
||||
'da' => ['name' => 'Danish', 'flag' => 'dk'],
|
||||
'ur' => ['name' => 'Urdu', 'flag' => 'pk'],
|
||||
'sq' => ['name' => 'Albanian', 'flag' => 'al'],
|
||||
'sk' => ['name' => 'Slovak', 'flag' => 'sk'],
|
||||
'bur' => ['name' => 'Burmese', 'flag' => 'mm'],
|
||||
'ti' => ['name' => 'Tigrinya', 'flag' => 'er'],
|
||||
'kz' => ['name' => 'Kazakh', 'flag' => 'kz'],
|
||||
'az' => ['name' => 'Azerbaijani', 'flag' => 'az'],
|
||||
'zh-cn' => ['name' => 'Chinese (CN)', 'flag' => 'zh-cn'],
|
||||
'zh-tw' => ['name' => 'Chinese (TW)', 'flag' => 'zh-tw'],
|
||||
'pt-br' => ['name' => 'Portuguese (BR)', 'flag' => 'pt-br'],
|
||||
'tz' => ['name' => 'Swahili', 'flag' => 'tz'],
|
||||
'ps' => ['name' => 'Pashto', 'flag' => 'af'],
|
||||
'prs' => ['name' => 'Dari', 'flag' => 'afdari'],
|
||||
'ca' => ['name' => 'Catalan', 'flag' => 'ad'],
|
||||
'bt' => ['name' => 'Dzongkha', 'flag' => 'dz'],
|
||||
'drcfr' => ['name' => 'Congo (DRC)', 'flag' => 'drc'],
|
||||
'cgfr' => ['name' => 'Congo (Republic)', 'flag' => 'cg'],
|
||||
'escr' => ['name' => 'Costa Rica (Spanish)', 'flag' => 'cr'],
|
||||
'enbw' => ['name' => 'Botswana (English)', 'flag' => 'bw'],
|
||||
'bws' => ['name' => 'Botswana (Setswana)', 'flag' => 'bws'],
|
||||
'deat' => ['name' => 'Austria(German)', 'flag' => 'at'],
|
||||
'enbs' => ['name' => 'Bahamas(English)', 'flag' => 'bs'],
|
||||
'arbh' => ['name' => 'Bahrain(Arabic)', 'flag' => 'bh'],
|
||||
'pt-ao' => ['name' => 'Angola(Portuguese)', 'flag' => 'ao'],
|
||||
'es-ar' => ['name' => 'Argentina(Spanish)', 'flag' => 'ar'],
|
||||
'hy' => ['name' => 'Armenian', 'flag' => 'am'],
|
||||
'au-en' => ['name' => 'Australia', 'flag' => 'au'],
|
||||
'bb-en' => ['name' => 'Barbados(English)', 'flag' => 'bb'],
|
||||
'be' => ['name' => 'Belarusian', 'flag' => 'by'],
|
||||
'nl-be' => ['name' => 'Belgium(Dutch)', 'flag' => 'be'],
|
||||
'bz-en' => ['name' => 'Belize(English)', 'flag' => 'bz'],
|
||||
'bj-fr' => ['name' => 'Benin(French)', 'flag' => 'bj'],
|
||||
'bo-es' => ['name' => 'Bolivia(Spanish)', 'flag' => 'bo'],
|
||||
'bn-ms' => ['name' => 'Brunei(Malay)', 'flag' => 'bn'],
|
||||
'bg' => ['name' => 'Bulgarian', 'flag' => 'bg'],
|
||||
'bf-fr' => ['name' => 'Burkina Faso(French)', 'flag' => 'bf'],
|
||||
'cm-fr' => ['name' => 'Cameroon(French)', 'flag' => 'cm'],
|
||||
'ca-en' => ['name' => 'Canada(English)', 'flag' => 'ca'],
|
||||
'cl-es' => ['name' => 'Chile(Spanish)', 'flag' => 'cl'],
|
||||
'co-es' => ['name' => 'Colombia(Spanish)', 'flag' => 'co'],
|
||||
'km-ar' => ['name' => 'Comoros(Arabic)', 'flag' => 'km'],
|
||||
'hr' => ['name' => 'Croatian', 'flag' => 'hr'],
|
||||
'cu-es' => ['name' => 'Cuba(Spanish)', 'flag' => 'cu'],
|
||||
'cy-el' => ['name' => 'Cyprus(Greek)', 'flag' => 'cy'],
|
||||
'dj-fr' => ['name' => 'Djibouti(French)', 'flag' => 'dj'],
|
||||
'dm-en' => ['name' => 'Dominica(English)', 'flag' => 'dm'],
|
||||
'tet' => ['name' => 'Tetum', 'flag' => 'tl'],
|
||||
'ec-es' => ['name' => 'Ecuador(Spanish)', 'flag' => 'ec'],
|
||||
'eg-ar' => ['name' => 'Egypt(Arabic)', 'flag' => 'eg'],
|
||||
'sv-es' => ['name' => 'El Salvador(Spanish)', 'flag' => 'sv'],
|
||||
'gq-es' => ['name' => 'Equatorial Guinea(Spanish)', 'flag' => 'gq'],
|
||||
'et' => ['name' => 'Estonian', 'flag' => 'ee'],
|
||||
'ss' => ['name' => 'Swati', 'flag' => 'sz'],
|
||||
'am' => ['name' => 'Amharic', 'flag' => 'et'],
|
||||
'fj' => ['name' => 'Fijian', 'flag' => 'fj'],
|
||||
'ga-fr' => ['name' => 'Gabon(French)', 'flag' => 'ga'],
|
||||
'gm-en' => ['name' => 'Gambia(English)', 'flag' => 'gm'],
|
||||
'ka' => ['name' => 'Georgian', 'flag' => 'ge'],
|
||||
'gh-en' => ['name' => 'Ghana(English)', 'flag' => 'gh'],
|
||||
'gd-en' => ['name' => 'Grenada(English)', 'flag' => 'gd'],
|
||||
'gt-en' => ['name' => 'Guatemala(English)', 'flag' => 'gt'],
|
||||
'gn-fr' => ['name' => 'Guinea(French)', 'flag' => 'gn'],
|
||||
'gy-en' => ['name' => 'Guyana(English)', 'flag' => 'gy'],
|
||||
'ht-fr' => ['name' => 'Haiti(French)', 'flag' => 'ht'],
|
||||
'hn-es' => ['name' => 'Honduras(Spanish)', 'flag' => 'hn'],
|
||||
];
|
||||
}
|
||||
|
||||
// BUSINESS PANEL
|
||||
|
||||
// user role permission
|
||||
if (!function_exists('visible_permission')) {
|
||||
function visible_permission($permission)
|
||||
{
|
||||
$user = auth()->user();
|
||||
|
||||
// Ensure the user is authenticated and has a business_id
|
||||
if (!$user || !$user->business_id) {
|
||||
return false;
|
||||
}
|
||||
|
||||
// Handle visibility field directly as an array or decode it if it's a string
|
||||
$permissions = is_array($user->visibility)
|
||||
? $user->visibility
|
||||
: json_decode($user->visibility, true);
|
||||
|
||||
return $permissions[$permission] ?? false;
|
||||
}
|
||||
}
|
||||
|
||||
function get_business_option($key)
|
||||
{
|
||||
$cacheKey = "business_setting_" . auth()->user()->business_id;
|
||||
|
||||
return Cache::remember($cacheKey, now()->addDay(), function () use ($key) {
|
||||
if ($key == 'business-settings') {
|
||||
$business_id = auth()->user()->business_id;
|
||||
|
||||
$option = Option::where('key', 'business-settings')
|
||||
->where('value', 'LIKE', '%"business_id":%' . $business_id . '%')
|
||||
->get()
|
||||
->firstWhere('value.business_id', $business_id);
|
||||
|
||||
return $option ? $option->value : null;
|
||||
}
|
||||
return null;
|
||||
});
|
||||
}
|
||||
|
||||
function plan_data($business_id = null)
|
||||
{
|
||||
$business_id = $business_id ?? auth()->user()->business_id;
|
||||
|
||||
return cache_remember('plan-data-' . $business_id, function () use ($business_id) {
|
||||
$planSubscribe = PlanSubscribe::with('plan:id,subscriptionName')->where('business_id', $business_id)->latest()->first();
|
||||
|
||||
if ($planSubscribe) {
|
||||
$business = Business::findOrFail($planSubscribe->business_id);
|
||||
$planSubscribe->will_expire = $business->will_expire;
|
||||
}
|
||||
return $planSubscribe;
|
||||
});
|
||||
}
|
||||
|
||||
function branch_count()
|
||||
{
|
||||
$business_id = auth()->user()->business_id;
|
||||
|
||||
return cache_remember('branch-count-' . $business_id, function () use ($business_id) {
|
||||
$totalBranch = Branch::where('business_id', $business_id)->count();
|
||||
|
||||
return $totalBranch;
|
||||
});
|
||||
}
|
||||
|
||||
function multibranch_active()
|
||||
{
|
||||
return plan_data()['allow_multibranch'] ?? false;
|
||||
}
|
||||
|
||||
function business_currency($business_id = null)
|
||||
{
|
||||
$business_id = $business_id ?? auth()->user()->business_id;
|
||||
|
||||
return cache_remember("business_currency_{$business_id}", function () use ($business_id) {
|
||||
$businessCurrency = UserCurrency::where('business_id', $business_id)->first() ?? Currency::where('is_default', 1)->first();
|
||||
;
|
||||
|
||||
if ($businessCurrency) {
|
||||
return (object) [
|
||||
'name' => $businessCurrency->name,
|
||||
'rate' => $businessCurrency->rate,
|
||||
'code' => $businessCurrency->code,
|
||||
'symbol' => $businessCurrency->symbol,
|
||||
'position' => $businessCurrency->position,
|
||||
];
|
||||
}
|
||||
|
||||
return default_currency();
|
||||
});
|
||||
}
|
||||
|
||||
function sale_rounding(?float $amount = null, ?string $round_option = null): float|string
|
||||
{
|
||||
$business_id = auth()->user()->business_id;
|
||||
|
||||
// If $round_option is not passed, try to fetch from settings
|
||||
if (is_null($round_option)) {
|
||||
$round_option = cache_remember("business_sale_rounding_{$business_id}", function () use ($business_id) {
|
||||
$option = Option::where('key', 'business-settings')
|
||||
->where('value', 'LIKE', '%"business_id":%' . $business_id . '%')
|
||||
->get()
|
||||
->firstWhere('value.business_id', $business_id);
|
||||
|
||||
return $option ? ($option->value['sale_rounding_option'] ?? 'none') : 'none';
|
||||
});
|
||||
}
|
||||
|
||||
if (is_null($amount)) {
|
||||
return $round_option;
|
||||
}
|
||||
|
||||
// Apply rounding if amount is provided
|
||||
return match ($round_option) {
|
||||
'round_up' => ceil($amount),
|
||||
'nearest_whole_number' => round($amount),
|
||||
'nearest_0.05' => round($amount * 20) / 20,
|
||||
'nearest_0.1' => round($amount * 10) / 10,
|
||||
'nearest_0.5' => round($amount * 2) / 2,
|
||||
default => $amount,
|
||||
};
|
||||
}
|
||||
|
||||
function moduleCheck($module)
|
||||
{
|
||||
$module = Module::find($module);
|
||||
|
||||
if ($module && $module->isEnabled()) {
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
function remaining_days($date)
|
||||
{
|
||||
$today = \Carbon\Carbon::today();
|
||||
$expiry = \Carbon\Carbon::parse($date);
|
||||
$diff = $today->diffInDays($expiry, false);
|
||||
|
||||
return $diff > 0 ? "$diff days" : "";
|
||||
}
|
||||
|
||||
// update RemainingBalance
|
||||
function updateBalance($amount, string $type, $branch_id = null)
|
||||
{
|
||||
$amount = is_numeric($amount) ? (float) $amount : 0;
|
||||
$businessId = auth()->user()->business_id;
|
||||
|
||||
// if active branch, then update active branch
|
||||
$branch = auth()->user()->active_branch;
|
||||
if ($branch) {
|
||||
if ($type == 'increment') {
|
||||
$branch->increment('branchRemainingBalance', $amount);
|
||||
} elseif ($type == 'decrement') {
|
||||
$branch->decrement('branchRemainingBalance', $amount);
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
//If branch_id is provided, update that branch
|
||||
if ($branch_id) {
|
||||
$branch = Branch::find($branch_id);
|
||||
if ($branch) {
|
||||
if ($type == 'increment') {
|
||||
$branch->increment('branchRemainingBalance', $amount);
|
||||
} elseif ($type == 'decrement') {
|
||||
$branch->decrement('branchRemainingBalance', $amount);
|
||||
}
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
// If no branch, update business balance
|
||||
$business = Business::find($businessId);
|
||||
if ($business) {
|
||||
if ($type == 'increment') {
|
||||
$business->increment('remainingShopBalance', $amount);
|
||||
} elseif ($type == 'decrement') {
|
||||
$business->decrement('remainingShopBalance', $amount);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
function manipulateBranchData($business_id)
|
||||
{
|
||||
$business = auth()->user()->business;
|
||||
$shop_owner = User::where(['business_id' => $business_id, 'role' => 'shop-owner'])->firstOrFail();
|
||||
|
||||
$branch = Branch::create([
|
||||
'is_main' => 1,
|
||||
'email' => $shop_owner->email,
|
||||
'name' => $business->companyName,
|
||||
'phone' => $business->phoneNumber,
|
||||
'address' => $business->address
|
||||
]);
|
||||
|
||||
$updates = [
|
||||
'users' => ['branch_id' => $branch->id, 'where' => ['role' => 'staff']],
|
||||
'stocks' => ['branch_id' => $branch->id, 'where' => ['business_id' => $business_id]],
|
||||
'product_settings' => ['branch_id' => $branch->id],
|
||||
'sale_returns' => ['branch_id' => $branch->id],
|
||||
'purchase_returns' => ['branch_id' => $branch->id],
|
||||
'expenses' => ['branch_id' => $branch->id],
|
||||
'incomes' => ['branch_id' => $branch->id],
|
||||
'sales' => ['branch_id' => $branch->id],
|
||||
'purchases' => ['branch_id' => $branch->id],
|
||||
'due_collects' => ['branch_id' => $branch->id],
|
||||
'parties' => ['branch_id' => $branch->id],
|
||||
'combo_products' => ['branch_id' => $branch->id],
|
||||
'transactions' => ['branch_id' => $branch->id],
|
||||
];
|
||||
|
||||
foreach ($updates as $table => $data) {
|
||||
$query = DB::table($table);
|
||||
if (!empty($data['where'])) {
|
||||
$query->where($data['where']);
|
||||
unset($data['where']);
|
||||
}
|
||||
$query->update($data);
|
||||
}
|
||||
|
||||
if (moduleCheck('HrmAddon')) {
|
||||
DB::table('holidays')->update(['branch_id' => $branch->id]);
|
||||
DB::table('attendances')->update(['branch_id' => $branch->id]);
|
||||
DB::table('leaves')->update(['branch_id' => $branch->id]);
|
||||
DB::table('payrolls')->update(['branch_id' => $branch->id]);
|
||||
DB::table('employees')->update(['branch_id' => $branch->id]);
|
||||
}
|
||||
|
||||
if (moduleCheck('WarehouseAddon')) {
|
||||
DB::table('warehouses')->update(['branch_id' => $branch->id]);
|
||||
DB::table('transfers')->update([
|
||||
'from_branch_id' => $branch->id,
|
||||
'to_branch_id' => $branch->id
|
||||
]);
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
function get_root_domain()
|
||||
{
|
||||
$appUrl = config('app.url');
|
||||
return parse_url($appUrl, PHP_URL_HOST);
|
||||
}
|
||||
|
||||
function checkDomainStatus($domain)
|
||||
{
|
||||
$result = [
|
||||
'domain' => $domain,
|
||||
'exists' => false,
|
||||
'http' => false,
|
||||
'https' => false,
|
||||
];
|
||||
|
||||
// 1. Check if domain resolves (DNS record exists)
|
||||
if (dns_get_record($domain, DNS_A) || dns_get_record($domain, DNS_AAAA)) {
|
||||
$result['exists'] = true;
|
||||
|
||||
// 2. Check HTTP (port 80)
|
||||
try {
|
||||
$response = Http::timeout(5)->get("http://{$domain}");
|
||||
if ($response->successful()) {
|
||||
$result['http'] = true;
|
||||
}
|
||||
} catch (\Exception $e) {
|
||||
$result['http'] = false;
|
||||
}
|
||||
|
||||
// 3. Check HTTPS (port 443)
|
||||
try {
|
||||
$response = Http::timeout(5)->get("https://{$domain}");
|
||||
if ($response->successful()) {
|
||||
$result['https'] = true;
|
||||
}
|
||||
} catch (\Exception $e) {
|
||||
$result['https'] = false;
|
||||
}
|
||||
}
|
||||
|
||||
return $result;
|
||||
}
|
||||
|
||||
function custom_reports()
|
||||
{
|
||||
$business_id = auth()->user()->business_id;
|
||||
|
||||
if (moduleCheck('CustomReportsAddon')) {
|
||||
return cache_remember('custom-reports-' . $business_id, function () use ($business_id) {
|
||||
return Modules\CustomReportsAddon\App\Models\CustomReport::where('business_id', auth()->user()->business_id)->where('status', 1)->get();
|
||||
});
|
||||
} else {
|
||||
return [];
|
||||
}
|
||||
}
|
||||
|
||||
function cash_balance()
|
||||
{
|
||||
$businessId = auth()->user()->business_id;
|
||||
|
||||
if (!$businessId) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
// Sum of all credits and bank_to_cash
|
||||
$totalCredit = Transaction::where('business_id', $businessId)
|
||||
->where(function ($q) {
|
||||
$q->where('type', 'credit')
|
||||
->orWhere('transaction_type', 'bank_to_cash');
|
||||
})
|
||||
->sum('amount');
|
||||
|
||||
// Sum of all debits and cash_to_bank
|
||||
$totalDebit = Transaction::where('business_id', $businessId)
|
||||
->where(function ($q) {
|
||||
$q->where('type', 'debit')
|
||||
->orWhere('transaction_type', 'cash_to_bank');
|
||||
})
|
||||
->sum('amount');
|
||||
|
||||
return $totalCredit - $totalDebit;
|
||||
}
|
||||
|
||||
function transaction_types($transactions): string
|
||||
{
|
||||
if (!$transactions) {
|
||||
return '';
|
||||
}
|
||||
|
||||
return $transactions
|
||||
->map(function ($transaction) {
|
||||
if (
|
||||
$transaction->transaction_type === 'bank_payment' &&
|
||||
!empty($transaction->paymentType?->name)
|
||||
) {
|
||||
return $transaction->paymentType->name;
|
||||
}
|
||||
|
||||
return $transaction->transaction_type
|
||||
? ucfirst(explode('_', $transaction->transaction_type)[0])
|
||||
: '';
|
||||
})
|
||||
->filter()
|
||||
->unique()
|
||||
->implode(', ');
|
||||
}
|
||||
Reference in New Issue
Block a user