Files
jongryangje/app/Controllers/Admin/DesignatedShop.php

1563 lines
61 KiB
PHP
Raw Normal View History

<?php
namespace App\Controllers\Admin;
use App\Controllers\BaseController;
use App\Models\CodeDetailModel;
use App\Models\CodeKindModel;
use App\Models\DesignatedShopModel;
use App\Models\LocalGovernmentModel;
use Config\Roles;
class DesignatedShop extends BaseController
{
private DesignatedShopModel $shopModel;
private LocalGovernmentModel $lgModel;
private Roles $roles;
public function __construct()
{
$this->shopModel = model(DesignatedShopModel::class);
$this->lgModel = model(LocalGovernmentModel::class);
$this->roles = config('Roles');
}
private function isSuperAdmin(): bool
{
return Roles::isSuperAdminEquivalent((int) session()->get('mb_level'));
}
private function isLocalAdmin(): bool
{
return (int) session()->get('mb_level') === Roles::LEVEL_LOCAL_ADMIN;
}
/**
* DB 행에서 컬럼이 없을 있는 환경( 스키마)까지 고려한 문자열 읽기.
*/
private function designatedShopScalar(object $row, string $field): string
{
return isset($row->{$field}) ? (string) $row->{$field} : '';
}
/**
* DATE 컬럼을 상세 JSON/화면용 문자열로.
*/
private function designatedShopDateOut(object $row, string $field): string
{
if (! isset($row->{$field})) {
return '';
}
$da = $row->{$field};
if ($da === null || $da === '' || $da === '0000-00-00') {
return '';
}
return (string) $da;
}
/**
* 가상계좌: 은행·계좌번호 분리 입력 + 단일 필드(ds_va_number) 하위 호환.
*
* @return array{ds_va_bank: string, ds_va_account: string, ds_va_number: string}
*/
private function resolveVirtualAccountFromRequest(): array
{
$bank = trim((string) $this->request->getPost('ds_va_bank'));
$account = trim((string) $this->request->getPost('ds_va_account'));
$legacy = trim((string) $this->request->getPost('ds_va_number'));
if ($account === '' && $legacy !== '') {
$account = $legacy;
}
$number = $account !== '' ? $account : $legacy;
return [
'ds_va_bank' => $bank,
'ds_va_account' => $account,
'ds_va_number' => $number,
];
}
private function normalizeOptionalDate(?string $raw): ?string
{
$s = trim((string) ($raw ?? ''));
if ($s === '' || $s === '0000-00-00') {
return null;
}
$dt = \DateTimeImmutable::createFromFormat('Y-m-d', $s);
return ($dt !== false && $dt->format('Y-m-d') === $s) ? $s : null;
}
/**
* 주소 문자열 비교용(공백 제거).
*/
private function compactAddressText(string $s): string
{
return preg_replace('/\s+/u', '', trim($s)) ?? '';
}
/**
* 시·도 또는 구·군 명칭이 지자체 마스터와 맞는지 본다.
* - 카카오 `sido`/`sigungu` 또는 도로명·지번 전체에 `lg_sido`·`lg_gugun` 포함되면 허용.
*/
private function koreanRegionTokenMatches(string $lgNeedle, string $primaryToken, string $fullBlob): bool
{
$needle = $this->compactAddressText($lgNeedle);
if ($needle === '') {
return true;
}
$blob = $this->compactAddressText($fullBlob);
if ($blob !== '' && mb_stripos($blob, $needle, 0, 'UTF-8') !== false) {
return true;
}
$primary = $this->compactAddressText($primaryToken);
if ($primary !== '') {
if (mb_stripos($primary, $needle, 0, 'UTF-8') !== false) {
return true;
}
if (mb_stripos($needle, $primary, 0, 'UTF-8') !== false) {
return true;
}
}
return false;
}
/**
* 우편·도로명·지번 하나라도 있으면, 효과 지자체(`lg_sido`, `lg_gugun`) 관할인지 검사한다.
*/
private function isDesignatedShopAddressWithinLocalGovernment(
object $lg,
string $addrSido,
string $addrSigungu,
string $road,
string $jibun,
string $zip
): bool {
$road = trim($road);
$jibun = trim($jibun);
$zip = trim($zip);
if ($road === '' && $jibun === '' && $zip === '') {
return true;
}
$lgSido = trim((string) ($lg->lg_sido ?? ''));
$lgGugun = trim((string) ($lg->lg_gugun ?? ''));
$blob = trim($addrSido . ' ' . $addrSigungu . ' ' . $road . ' ' . $jibun . ' ' . $zip);
if ($lgSido !== '' && ! $this->koreanRegionTokenMatches($lgSido, $addrSido, $blob)) {
return false;
}
if ($lgGugun !== '' && ! $this->koreanRegionTokenMatches($lgGugun, $addrSigungu, $blob)) {
return false;
}
return true;
}
private function hasDesignatedShopPostalAddress(string $zip, string $road, string $jibun): bool
{
return trim($zip . $road . $jibun) !== '';
}
/**
* 우편·도로명·지번이 있으면 카카오 검색으로만 채웠는지(시도 hidden) 확인한다.
*/
private function designatedShopAddressFilledWithoutSearch(string $addrSido, string $zip, string $road, string $jibun): bool
{
return $this->hasDesignatedShopPostalAddress($zip, $road, $jibun) && trim($addrSido) === '';
}
/**
* 목록 검색과 동일한 조건을 모델 쿼리에 적용한다.
*/
private function applyDesignatedShopListFilters(DesignatedShopModel $model, int $lgIdx, ?string $dsName, ?string $dsGugunCode, ?string $dsState): void
{
$model->where('ds_lg_idx', $lgIdx);
if ($dsName !== null && $dsName !== '') {
$model->like('ds_name', $dsName);
}
if ($dsGugunCode !== null && $dsGugunCode !== '') {
$model->where('ds_gugun_code', $dsGugunCode);
}
if ($dsState !== null && $dsState !== '') {
$model->where('ds_state', (int) $dsState);
}
}
/**
* @return array{1: int, 2: int, 3: int, total: int}
*/
private function countDesignatedShopsByState(int $lgIdx, ?string $dsName, ?string $dsGugunCode, ?string $dsState): array
{
$db = \Config\Database::connect();
$builder = $db->table('designated_shop');
$builder->where('ds_lg_idx', $lgIdx);
if ($dsName !== null && $dsName !== '') {
$builder->like('ds_name', $dsName);
}
if ($dsGugunCode !== null && $dsGugunCode !== '') {
$builder->where('ds_gugun_code', $dsGugunCode);
}
if ($dsState !== null && $dsState !== '') {
$builder->where('ds_state', (int) $dsState);
}
$rows = $builder->select('ds_state, COUNT(*) AS cnt', false)
->groupBy('ds_state')
->get()
->getResultArray();
$counts = [1 => 0, 2 => 0, 3 => 0];
foreach ($rows as $r) {
$st = (int) ($r['ds_state'] ?? 0);
if (isset($counts[$st])) {
$counts[$st] = (int) $r['cnt'];
}
}
$counts['total'] = $counts[1] + $counts[2] + $counts[3];
return $counts;
}
/**
* @param list<object> $list
* @param array<int, string> $lgMap
* @return list<array<string, mixed>>
*/
private function buildDesignatedShopDetailPayload(array $list, array $lgMap): array
{
$payload = [];
foreach ($list as $row) {
$sn = (string) ($row->ds_shop_no ?? '');
if (preg_match('/(\d{3})$/', $sn, $m)) {
$shortNo = $m[1];
} elseif ($sn !== '' && strlen($sn) >= 3) {
$shortNo = substr($sn, -3);
} else {
$shortNo = $sn;
}
$st = (int) ($row->ds_state ?? 1);
$stateMap = [1 => '정상', 2 => '폐업', 3 => '직권해지'];
$da = $row->ds_designated_at ?? null;
$daOut = ($da !== null && $da !== '' && $da !== '0000-00-00') ? (string) $da : '';
$payload[] = [
'ds_idx' => (int) $row->ds_idx,
'ds_shop_no' => $sn,
'shop_no_display' => $shortNo,
'ds_rep_name' => (string) ($row->ds_rep_name ?? ''),
'ds_name' => (string) ($row->ds_name ?? ''),
'ds_state' => $st,
'state_label' => $stateMap[$st] ?? '',
'ds_biz_no' => (string) ($row->ds_biz_no ?? ''),
'ds_biz_type' => $this->designatedShopScalar($row, 'ds_biz_type'),
'ds_biz_kind' => $this->designatedShopScalar($row, 'ds_biz_kind'),
'ds_va_number' => (string) ($row->ds_va_number ?? ''),
'ds_va_bank' => $this->designatedShopScalar($row, 'ds_va_bank'),
'ds_va_account' => $this->designatedShopScalar($row, 'ds_va_account'),
'ds_zip' => (string) ($row->ds_zip ?? ''),
'ds_addr' => (string) ($row->ds_addr ?? ''),
'ds_addr_jibun' => (string) ($row->ds_addr_jibun ?? ''),
'ds_addr_detail' => $this->designatedShopScalar($row, 'ds_addr_detail'),
'ds_tel' => (string) ($row->ds_tel ?? ''),
'ds_rep_phone' => (string) ($row->ds_rep_phone ?? ''),
'ds_email' => (string) ($row->ds_email ?? ''),
'ds_gugun_code' => (string) ($row->ds_gugun_code ?? ''),
'ds_zone_code' => $this->designatedShopScalar($row, 'ds_zone_code'),
'ds_branch_no' => $this->designatedShopScalar($row, 'ds_branch_no'),
'ds_designated_at' => $daOut,
'ds_state_changed_at' => $this->designatedShopDateOut($row, 'ds_state_changed_at'),
'ds_change_reason' => $this->designatedShopScalar($row, 'ds_change_reason'),
'ds_regdate' => (string) ($row->ds_regdate ?? ''),
'lg_name' => $lgMap[(int) ($row->ds_lg_idx ?? 0)] ?? '',
];
}
return $payload;
}
/**
* 목록·검색·상세 표시에 공통으로 쓰는 데이터 (지자체 미선택 null).
*
* @return array<string, mixed>|null
*/
private function designatedShopIndexViewData(): ?array
{
helper('admin');
$lgIdx = admin_effective_lg_idx();
if ($lgIdx === null || $lgIdx <= 0) {
return null;
}
// 다조건 검색 (P2-15)
$dsName = $this->request->getGet('ds_name');
$dsGugunCode = $this->request->getGet('ds_gugun_code');
$dsState = $this->request->getGet('ds_state');
$this->applyDesignatedShopListFilters($this->shopModel, $lgIdx, $dsName, $dsGugunCode, $dsState);
$list = $this->shopModel->orderBy('ds_idx', 'DESC')->paginate(20);
$pager = $this->shopModel->pager;
// 지자체 이름 매핑용
$lgMap = [];
foreach ($this->lgModel->findAll() as $lg) {
$lgMap[$lg->lg_idx] = $lg->lg_name;
}
$stateCounts = $this->countDesignatedShopsByState($lgIdx, $dsName, $dsGugunCode, $dsState);
$detailRows = $this->buildDesignatedShopDetailPayload($list, $lgMap);
// 구군코드 목록 (검색 필터용)
$db = \Config\Database::connect();
$gugunCodes = $db->query("SELECT DISTINCT ds_gugun_code FROM designated_shop WHERE ds_lg_idx = ? AND ds_gugun_code != '' ORDER BY ds_gugun_code", [$lgIdx])->getResult();
return [
'list' => $list,
'lgMap' => $lgMap,
'pager' => $pager,
'dsName' => $dsName ?? '',
'dsGugunCode' => $dsGugunCode ?? '',
'dsState' => $dsState ?? '',
'gugunCodes' => $gugunCodes,
'stateCounts' => $stateCounts,
'detailRowsJson' => json_encode($detailRows, JSON_HEX_TAG | JSON_HEX_APOS | JSON_HEX_QUOT | JSON_UNESCAPED_UNICODE),
'kakaoJavascriptKey' => $this->kakaoJavascriptKey(),
];
}
/**
* 지정판매소 목록 (효과 지자체 기준: super admin = 선택 지자체, 지자체관리자 = mb_lg_idx)
*/
public function index()
{
$data = $this->designatedShopIndexViewData();
if ($data === null) {
return redirect()->to(work_area_home_url())
->with('error', '작업할 지자체가 선택되지 않았습니다. 지자체를 선택해 주세요.');
}
return $this->renderWorkPage('지정판매소 관리', 'admin/designated_shop/index', $data);
}
/**
* 지정판매소 조회 전용 (목록·검색·상세만, 등록·수정·삭제·엑셀 없음)
*/
public function browse()
{
$data = $this->designatedShopIndexViewData();
if ($data === null) {
return redirect()->to(work_area_home_url())
->with('error', '작업할 지자체가 선택되지 않았습니다. 지자체를 선택해 주세요.');
}
$data['readOnly'] = true;
return $this->renderWorkPage('지정판매소 조회', 'admin/designated_shop/index', $data);
}
/**
* 지정판매소 바코드 출력 전용 목록 (선택 인쇄).
*/
public function barcode()
{
helper('admin');
$lgIdx = admin_effective_lg_idx();
if ($lgIdx === null || $lgIdx <= 0) {
return redirect()->to(work_area_home_url())
->with('error', '작업할 지자체가 선택되지 않았습니다. 지자체를 선택해 주세요.');
}
$currentLg = $this->lgModel->find($lgIdx);
$fixedGugunCode = trim((string) ($currentLg->lg_code ?? ''));
$fixedGugunLabel = trim((string) ($currentLg->lg_gugun ?? ''));
if ($fixedGugunLabel === '') {
$fixedGugunLabel = trim((string) ($currentLg->lg_name ?? ''));
}
$zone = trim((string) $this->request->getGet('ds_zone_code'));
$order = trim((string) $this->request->getGet('order_by'));
if (! in_array($order, ['shop_no', 'name'], true)) {
$order = 'shop_no';
}
$builder = $this->shopModel->where('ds_lg_idx', $lgIdx);
if ($fixedGugunCode !== '') {
$builder->where('ds_gugun_code', $fixedGugunCode);
}
if ($zone !== '') {
$builder->where('ds_zone_code', $zone);
}
if ($order === 'name') {
$builder->orderBy('ds_name', 'ASC');
} else {
$builder->orderBy('ds_shop_no', 'ASC');
}
$builder->orderBy('ds_idx', 'ASC');
$list = $builder->paginate(100);
$db = \Config\Database::connect();
$zones = $db->query(
"SELECT DISTINCT TRIM(ds_zone_code) AS zone_code
FROM designated_shop
WHERE ds_lg_idx = ?
AND (? = '' OR ds_gugun_code = ?)
AND TRIM(ds_zone_code) != ''
ORDER BY zone_code",
[$lgIdx, $fixedGugunCode, $fixedGugunCode]
)->getResult();
return $this->renderWorkPage('지정판매소 바코드 출력', 'admin/designated_shop/barcode', [
'list' => $list,
'pager' => $this->shopModel->pager,
'fixedGugunLabel' => $fixedGugunLabel,
'zoneFilter' => $zone,
'zones' => $zones,
'orderBy' => $order,
]);
}
/**
* 지정판매소 바코드 인쇄 페이지 (선택된 판매소 기준).
*/
public function barcodePrint()
{
helper('admin');
$lgIdx = admin_effective_lg_idx();
if ($lgIdx === null || $lgIdx <= 0) {
return redirect()->to(work_area_home_url())
->with('error', '작업할 지자체가 선택되지 않았습니다. 지자체를 선택해 주세요.');
}
$ids = $this->request->getPost('ds_idx');
$ids = is_array($ids) ? array_values(array_unique(array_map('intval', $ids))) : [];
$ids = array_values(array_filter($ids, static fn ($v): bool => $v > 0));
if ($ids === []) {
return redirect()->to(mgmt_url('designated-shops/barcode'))
->with('error', '출력할 지정판매소를 선택해 주세요.');
}
$rows = $this->shopModel
->where('ds_lg_idx', $lgIdx)
->whereIn('ds_idx', $ids)
->orderBy('ds_shop_no', 'ASC')
->findAll();
if ($rows === []) {
return redirect()->to(mgmt_url('designated-shops/barcode'))
->with('error', '선택한 지정판매소를 찾을 수 없습니다.');
}
$zoneLabel = trim((string) $this->request->getPost('zone_label'));
if ($zoneLabel === '') {
$firstZone = trim((string) ($rows[0]->ds_zone_code ?? ''));
$zoneLabel = $firstZone !== '' ? $firstZone : '전체';
}
return view('admin/designated_shop/barcode_print', [
'rows' => $rows,
'zoneLabel' => $zoneLabel,
'printedAt' => date('Y.m.d'),
'totalCount' => count($rows),
]);
}
public function export()
{
helper(['admin', 'export']);
$lgIdx = admin_effective_lg_idx();
if (!$lgIdx) {
return redirect()->to(mgmt_url('designated-shops'))->with('error', '지자체를 선택해 주세요.');
}
$list = $this->shopModel->where('ds_lg_idx', $lgIdx)->orderBy('ds_idx', 'DESC')->findAll();
$rows = [];
foreach ($list as $row) {
$stateMap = [1 => '정상', 2 => '폐업', 3 => '직권해지'];
$rows[] = [
$row->ds_idx,
$row->ds_shop_no,
$row->ds_name,
$row->ds_rep_name,
$row->ds_biz_no,
$this->designatedShopScalar($row, 'ds_biz_type'),
$this->designatedShopScalar($row, 'ds_biz_kind'),
$this->designatedShopScalar($row, 'ds_va_bank'),
$this->designatedShopScalar($row, 'ds_va_account') !== '' ? $this->designatedShopScalar($row, 'ds_va_account') : ($row->ds_va_number ?? ''),
$row->ds_tel ?? '',
$row->ds_addr ?? '',
$this->designatedShopScalar($row, 'ds_zone_code'),
$this->designatedShopScalar($row, 'ds_branch_no'),
$this->designatedShopDateOut($row, 'ds_state_changed_at'),
$this->designatedShopScalar($row, 'ds_change_reason'),
$stateMap[(int) $row->ds_state] ?? '',
$row->ds_regdate ?? '',
];
}
export_csv(
'지정판매소_' . date('Ymd') . '.csv',
[
'번호', '판매소번호', '상호명', '대표자', '사업자번호', '업태', '업종',
'가상계좌은행', '계좌번호', '전화번호', '주소', '구역', '종사업장번호',
'변경일자', '변경사유', '상태', '등록일',
],
$rows
);
}
/**
* 지정판매소 등록 (효과 지자체 기준)
*/
public function create()
{
helper('admin');
$lgIdx = admin_effective_lg_idx();
if ($lgIdx === null || $lgIdx <= 0) {
return redirect()->to(mgmt_url('designated-shops'))
->with('error', '작업할 지자체가 선택되지 않았습니다. 지자체를 선택해 주세요.');
}
$currentLg = $this->lgModel->find($lgIdx);
if ($currentLg === null) {
return redirect()->to(mgmt_url('designated-shops'))
->with('error', '선택한 지자체 정보를 찾을 수 없습니다.');
}
return $this->renderWorkPage('지정판매소 등록', 'admin/designated_shop/create', [
'localGovs' => [],
'currentLg' => $currentLg,
'addrTenantScope' => [
'lg_sido' => (string) ($currentLg->lg_sido ?? ''),
'lg_gugun' => (string) ($currentLg->lg_gugun ?? ''),
],
'kakaoJavascriptKey' => $this->kakaoJavascriptKey(),
]);
}
/**
* 지정판매소 등록 처리
*/
public function store()
{
if (! $this->isSuperAdmin() && ! $this->isLocalAdmin()) {
return redirect()->to(mgmt_url('designated-shops'))
->with('error', '지정판매소 등록은 관리자만 가능합니다.');
}
$rules = [
'ds_name' => 'required|max_length[100]',
'ds_biz_no' => 'required|max_length[20]',
'ds_rep_name' => 'required|max_length[50]',
'ds_biz_type' => 'permit_empty|max_length[100]',
'ds_biz_kind' => 'permit_empty|max_length[100]',
'ds_va_number' => 'permit_empty|max_length[50]',
'ds_va_bank' => 'permit_empty|max_length[80]',
'ds_va_account' => 'permit_empty|max_length[50]',
'ds_email' => 'permit_empty|valid_email|max_length[100]',
'ds_zone_code' => 'permit_empty|max_length[80]',
'ds_branch_no' => 'permit_empty|max_length[50]',
'ds_change_reason' => 'permit_empty|max_length[500]',
'ds_state_changed_at' => 'permit_empty|max_length[10]',
'addr_search_sido' => 'permit_empty|max_length[80]',
'addr_search_sigungu' => 'permit_empty|max_length[80]',
'ds_addr_detail' => 'permit_empty|max_length[200]',
];
if (! $this->validate($rules)) {
return redirect()->back()
->withInput()
->with('errors', $this->validator->getErrors());
}
helper('admin');
$lgIdx = admin_effective_lg_idx();
if ($lgIdx === null || $lgIdx <= 0) {
return redirect()->back()
->withInput()
->with('error', '소속 지자체가 올바르지 않습니다.');
}
$lg = $this->lgModel->find($lgIdx);
if ($lg === null || (string) $lg->lg_code === '') {
return redirect()->back()
->withInput()
->with('error', '지자체 코드 정보를 찾을 수 없습니다.');
}
$addrSido = (string) $this->request->getPost('addr_search_sido');
$addrSigungu = (string) $this->request->getPost('addr_search_sigungu');
$dsAddr = (string) $this->request->getPost('ds_addr');
$dsAddrJibun = (string) $this->request->getPost('ds_addr_jibun');
$dsZip = (string) $this->request->getPost('ds_zip');
if ($this->designatedShopAddressFilledWithoutSearch($addrSido, $dsZip, $dsAddr, $dsAddrJibun)) {
return redirect()->back()
->withInput()
->with('error', '주소는 「주소 검색」으로만 지정할 수 있습니다.');
}
if (! $this->isDesignatedShopAddressWithinLocalGovernment($lg, $addrSido, $addrSigungu, $dsAddr, $dsAddrJibun, $dsZip)) {
return redirect()->back()
->withInput()
->with('error', '작업 중인 지자체(' . (string) $lg->lg_name . ') 관할 주소만 등록할 수 있습니다. 주소 검색으로 선택하거나 시·구에 맞게 입력해 주세요.');
}
$resolvedNo = $this->resolveDesignatedShopNumberFromAddress(
$lgIdx,
$addrSido,
$addrSigungu,
$dsAddr,
$dsAddrJibun,
$dsZip,
$lg
);
if (! $resolvedNo['ok']) {
return redirect()->back()
->withInput()
->with('error', $resolvedNo['error']);
}
$va = $this->resolveVirtualAccountFromRequest();
$data = [
'ds_lg_idx' => $lgIdx,
'ds_mb_idx' => null,
'ds_shop_no' => $resolvedNo['shop_no'],
'ds_name' => (string) $this->request->getPost('ds_name'),
'ds_biz_no' => (string) $this->request->getPost('ds_biz_no'),
'ds_rep_name' => (string) $this->request->getPost('ds_rep_name'),
'ds_biz_type' => (string) $this->request->getPost('ds_biz_type'),
'ds_biz_kind' => (string) $this->request->getPost('ds_biz_kind'),
'ds_va_bank' => $va['ds_va_bank'],
'ds_va_account' => $va['ds_va_account'],
'ds_va_number' => $va['ds_va_number'],
'ds_zip' => (string) $this->request->getPost('ds_zip'),
'ds_addr' => (string) $this->request->getPost('ds_addr'),
'ds_addr_jibun' => (string) $this->request->getPost('ds_addr_jibun'),
'ds_addr_detail' => (string) $this->request->getPost('ds_addr_detail'),
'ds_tel' => (string) $this->request->getPost('ds_tel'),
'ds_rep_phone' => (string) $this->request->getPost('ds_rep_phone'),
'ds_email' => (string) $this->request->getPost('ds_email'),
'ds_gugun_code' => $resolvedNo['gugun_code'],
'ds_zone_code' => (string) $this->request->getPost('ds_zone_code'),
'ds_branch_no' => (string) $this->request->getPost('ds_branch_no'),
'ds_designated_at' => $this->request->getPost('ds_designated_at') ?: null,
'ds_state' => 1,
'ds_state_changed_at' => $this->normalizeOptionalDate($this->request->getPost('ds_state_changed_at')),
'ds_change_reason' => (string) $this->request->getPost('ds_change_reason'),
'ds_regdate' => date('Y-m-d H:i:s'),
];
$this->shopModel->insert($data);
return redirect()->to(mgmt_url('designated-shops'))
->with('success', '지정판매소가 등록되었습니다.');
}
/**
* 지정판매소 수정 (효과 지자체 소속만 허용)
* 문서: docs/기본 개발계획/23-지정판매소_수정_삭제_기능.md
*/
public function edit(int $id)
{
if (! $this->isSuperAdmin() && ! $this->isLocalAdmin()) {
return redirect()->to(mgmt_url('designated-shops'))
->with('error', '권한이 없습니다.');
}
helper('admin');
$lgIdx = admin_effective_lg_idx();
if ($lgIdx === null || $lgIdx <= 0) {
return redirect()->to(mgmt_url('designated-shops'))
->with('error', '작업할 지자체가 선택되지 않았습니다.');
}
$shop = $this->shopModel->find($id);
if ($shop === null || (int) $shop->ds_lg_idx !== $lgIdx) {
return redirect()->to(mgmt_url('designated-shops'))
->with('error', '해당 지정판매소를 찾을 수 없거나 수정할 수 없습니다.');
}
$currentLg = $this->lgModel->find($lgIdx);
return $this->renderWorkPage('지정판매소 수정', 'admin/designated_shop/edit', [
'shop' => $shop,
'currentLg' => $currentLg,
'addrTenantScope' => $currentLg !== null ? [
'lg_sido' => (string) ($currentLg->lg_sido ?? ''),
'lg_gugun' => (string) ($currentLg->lg_gugun ?? ''),
] : ['lg_sido' => '', 'lg_gugun' => ''],
'kakaoJavascriptKey' => $this->kakaoJavascriptKey(),
]);
}
/**
* 지정판매소 수정 처리
*/
public function update(int $id)
{
if (! $this->isSuperAdmin() && ! $this->isLocalAdmin()) {
return redirect()->to(mgmt_url('designated-shops'))
->with('error', '권한이 없습니다.');
}
helper('admin');
$lgIdx = admin_effective_lg_idx();
if ($lgIdx === null || $lgIdx <= 0) {
return redirect()->to(mgmt_url('designated-shops'))
->with('error', '작업할 지자체가 선택되지 않았습니다.');
}
$shop = $this->shopModel->find($id);
if ($shop === null || (int) $shop->ds_lg_idx !== $lgIdx) {
return redirect()->to(mgmt_url('designated-shops'))
->with('error', '해당 지정판매소를 찾을 수 없거나 수정할 수 없습니다.');
}
$rules = [
'ds_name' => 'required|max_length[100]',
'ds_biz_no' => 'required|max_length[20]',
'ds_rep_name' => 'required|max_length[50]',
'ds_biz_type' => 'permit_empty|max_length[100]',
'ds_biz_kind' => 'permit_empty|max_length[100]',
'ds_va_number' => 'permit_empty|max_length[50]',
'ds_va_bank' => 'permit_empty|max_length[80]',
'ds_va_account' => 'permit_empty|max_length[50]',
'ds_email' => 'permit_empty|valid_email|max_length[100]',
'ds_state' => 'permit_empty|in_list[1,2,3]',
'ds_zone_code' => 'permit_empty|max_length[80]',
'ds_branch_no' => 'permit_empty|max_length[50]',
'ds_change_reason' => 'permit_empty|max_length[500]',
'ds_state_changed_at' => 'permit_empty|max_length[10]',
'addr_search_sido' => 'permit_empty|max_length[80]',
'addr_search_sigungu' => 'permit_empty|max_length[80]',
'ds_addr_detail' => 'permit_empty|max_length[200]',
];
if (! $this->validate($rules)) {
return redirect()->back()
->withInput()
->with('errors', $this->validator->getErrors());
}
$lg = $this->lgModel->find($lgIdx);
if ($lg === null) {
return redirect()->back()
->withInput()
->with('error', '지자체 정보를 찾을 수 없습니다.');
}
$addrSido = (string) $this->request->getPost('addr_search_sido');
$addrSigungu = (string) $this->request->getPost('addr_search_sigungu');
$dsAddr = (string) $this->request->getPost('ds_addr');
$dsAddrJibun = (string) $this->request->getPost('ds_addr_jibun');
$dsZip = (string) $this->request->getPost('ds_zip');
if ($this->designatedShopAddressFilledWithoutSearch($addrSido, $dsZip, $dsAddr, $dsAddrJibun)) {
return redirect()->back()
->withInput()
->with('error', '주소는 「주소 검색」으로만 지정할 수 있습니다.');
}
if (! $this->isDesignatedShopAddressWithinLocalGovernment($lg, $addrSido, $addrSigungu, $dsAddr, $dsAddrJibun, $dsZip)) {
return redirect()->back()
->withInput()
->with('error', '작업 중인 지자체(' . (string) $lg->lg_name . ') 관할 주소만 입력할 수 있습니다. 주소 검색으로 선택하거나 시·구에 맞게 입력해 주세요.');
}
$va = $this->resolveVirtualAccountFromRequest();
$data = [
'ds_name' => (string) $this->request->getPost('ds_name'),
'ds_biz_no' => (string) $this->request->getPost('ds_biz_no'),
'ds_rep_name' => (string) $this->request->getPost('ds_rep_name'),
'ds_biz_type' => (string) $this->request->getPost('ds_biz_type'),
'ds_biz_kind' => (string) $this->request->getPost('ds_biz_kind'),
'ds_va_bank' => $va['ds_va_bank'],
'ds_va_account' => $va['ds_va_account'],
'ds_va_number' => $va['ds_va_number'],
'ds_zip' => (string) $this->request->getPost('ds_zip'),
'ds_addr' => (string) $this->request->getPost('ds_addr'),
'ds_addr_jibun' => (string) $this->request->getPost('ds_addr_jibun'),
'ds_addr_detail' => (string) $this->request->getPost('ds_addr_detail'),
'ds_tel' => (string) $this->request->getPost('ds_tel'),
'ds_rep_phone' => (string) $this->request->getPost('ds_rep_phone'),
'ds_email' => (string) $this->request->getPost('ds_email'),
'ds_zone_code' => (string) $this->request->getPost('ds_zone_code'),
'ds_branch_no' => (string) $this->request->getPost('ds_branch_no'),
'ds_designated_at' => $this->request->getPost('ds_designated_at') ?: null,
'ds_state' => (int) ($this->request->getPost('ds_state') ?: 1),
'ds_state_changed_at' => $this->normalizeOptionalDate($this->request->getPost('ds_state_changed_at')),
'ds_change_reason' => (string) $this->request->getPost('ds_change_reason'),
];
$this->shopModel->update($id, $data);
return redirect()->to(mgmt_url('designated-shops'))
->with('success', '지정판매소 정보가 수정되었습니다.');
}
/**
* 지정판매소 삭제 (물리 삭제, 효과 지자체 소속만 허용)
* 문서: docs/기본 개발계획/23-지정판매소_수정_삭제_기능.md
*/
public function delete(int $id)
{
if (! $this->isSuperAdmin() && ! $this->isLocalAdmin()) {
return redirect()->to(mgmt_url('designated-shops'))
->with('error', '권한이 없습니다.');
}
helper('admin');
$lgIdx = admin_effective_lg_idx();
if ($lgIdx === null || $lgIdx <= 0) {
return redirect()->to(mgmt_url('designated-shops'))
->with('error', '작업할 지자체가 선택되지 않았습니다.');
}
$shop = $this->shopModel->find($id);
if ($shop === null || (int) $shop->ds_lg_idx !== $lgIdx) {
return redirect()->to(mgmt_url('designated-shops'))
->with('error', '해당 지정판매소를 찾을 수 없거나 삭제할 수 없습니다.');
}
$this->shopModel->delete($id);
return redirect()->to(mgmt_url('designated-shops'))
->with('success', '지정판매소가 삭제되었습니다.');
}
/**
* P2-17: 지정판매소 지도 표시
*/
public function map()
{
helper('admin');
$lgIdx = admin_effective_lg_idx();
if ($lgIdx === null || $lgIdx <= 0) {
return redirect()->to(work_area_home_url())
->with('error', '작업할 지자체가 선택되지 않았습니다.');
}
$shops = $this->shopModel
->where('ds_lg_idx', $lgIdx)
->where('ds_state', 1)
->findAll();
return $this->renderWorkPage('지정판매소 지도', 'admin/designated_shop/map', [
'shops' => $shops,
'kakaoJavascriptKey' => $this->kakaoJavascriptKey(),
]);
}
/**
* 구·군 코드 표시명 (코드종류 C, 플랫폼+지자체 범위).
*
* @return array<string, string>
*/
private function gugunCodeNameMap(int $lgIdx): array
{
$ckIdx = $this->codeKindIdxByCkCode('C');
if ($ckIdx === null) {
return [];
}
$rows = model(CodeDetailModel::class)->getByKind($ckIdx, true, $lgIdx);
$map = [];
foreach ($rows as $r) {
$map[(string) $r->cd_code] = (string) $r->cd_name;
}
return $map;
}
/**
* 효과 지자체 기준 구·군 마스터 (기본코드 C 없으면 지정판매소에 실제 입력된 코드).
*
* @return list<array{code: string, name: string}>
*/
private function gugunMasterRowsForLg(int $lgIdx): array
{
$map = $this->gugunCodeNameMap($lgIdx);
$rows = [];
foreach ($map as $code => $name) {
$code = trim((string) $code);
if ($code === '') {
continue;
}
$rows[] = ['code' => $code, 'name' => trim((string) $name)];
}
usort($rows, static fn (array $a, array $b): int => strcmp($a['code'], $b['code']));
if ($rows !== []) {
return $rows;
}
$db = \Config\Database::connect();
$q = $db->query(
"SELECT DISTINCT TRIM(ds_gugun_code) AS c FROM designated_shop WHERE ds_lg_idx = ? AND TRIM(ds_gugun_code) != '' ORDER BY c",
[$lgIdx]
)->getResult();
foreach ($q as $o) {
$c = trim((string) ($o->c ?? ''));
if ($c === '') {
continue;
}
$rows[] = ['code' => $c, 'name' => $c];
}
return $rows;
}
/**
* GBMS형: 마스터 구·군 순서로 행을 채우고, 집계는 designatedShopDistrictStatusRows(구군 전체) 병합.
*
* @return array{rows: list<object>, total: object}
*/
private function buildGbmsNewCancelRows(int $lgIdx, int $year): array
{
$byCode = [];
foreach ($this->designatedShopDistrictStatusRows($lgIdx, $year, '', 'gugun') as $r) {
$byCode[(string) $r->gugun_code] = $r;
}
$master = $this->gugunMasterRowsForLg($lgIdx);
$out = [];
$seen = [];
foreach ($master as $m) {
$c = $m['code'];
$seen[$c] = true;
$st = $byCode[$c] ?? null;
$out[] = $this->makeGbmsDistrictRow($m['name'], $c, $st);
}
foreach ($byCode as $c => $st) {
if ($c === '' || isset($seen[$c])) {
continue;
}
$out[] = $this->makeGbmsDistrictRow((string) $st->region_label, $c, $st);
$seen[$c] = true;
}
if (isset($byCode[''])) {
$st = $byCode[''];
$out[] = $this->makeGbmsDistrictRow('(구·군 미입력)', '', $st);
}
if ($out === [] && $byCode !== []) {
foreach ($byCode as $c => $st) {
$out[] = $this->makeGbmsDistrictRow((string) $st->region_label, $c, $st);
}
}
$sumP = $sumD = $sumC = $sumCur = 0;
foreach ($out as $o) {
$sumP += (int) $o->prev_end;
$sumD += (int) $o->designated_y;
$sumC += (int) $o->cancelled_y;
$sumCur += (int) $o->curr_end;
}
return [
'rows' => $out,
'total' => (object) [
'region_label' => '계',
'prev_end' => $sumP,
'designated_y' => $sumD,
'cancelled_y' => $sumC,
'curr_end' => $sumCur,
],
];
}
/**
* @param object|null $st designatedShopDistrictStatusRows 또는 null
*/
private function makeGbmsDistrictRow(string $displayLabel, string $gugunCode, ?object $st): object
{
return (object) [
'region_label' => $displayLabel,
'gugun_code' => $gugunCode,
'prev_end' => $st !== null ? (int) $st->prev_end : 0,
'designated_y' => $st !== null ? (int) $st->designated_y : 0,
'cancelled_y' => $st !== null ? (int) $st->cancelled_y : 0,
'curr_end' => $st !== null ? (int) $st->curr_end : 0,
];
}
/**
* 연도·구군·집계 단위별 신규/취소 현황 (종전=전년말, 지정·취소=당해, 현행=금년말).
*
* @return list<object{region_key: string, region_label: string, prev_end: int, designated_y: int, cancelled_y: int, curr_end: int}>
*/
private function designatedShopDistrictStatusRows(
int $lgIdx,
int $year,
string $filterGugunCode,
string $granularity
): array {
$year = max(1990, min(2100, $year));
$prevEnd = sprintf('%04d-12-31', $year - 1);
$currEnd = sprintf('%04d-12-31', $year);
$granularity = $granularity === 'dong' ? 'dong' : 'gugun';
$db = \Config\Database::connect();
$gugunMap = $this->gugunCodeNameMap($lgIdx);
$filterGugunCode = trim($filterGugunCode);
$bind = [
$prevEnd,
$prevEnd,
$year,
$year,
$currEnd,
$currEnd,
$lgIdx,
$filterGugunCode,
$filterGugunCode,
];
if ($granularity === 'dong') {
$sql = "
SELECT
IFNULL(NULLIF(TRIM(ds_gugun_code), ''), '') AS gugun_key,
IFNULL(NULLIF(TRIM(ds_zone_code), ''), '') AS zone_key,
SUM(
CASE WHEN COALESCE(ds_designated_at, DATE(ds_regdate)) <= ?
AND (ds_state = 1 OR (ds_state IN (2,3) AND COALESCE(ds_state_changed_at, DATE(ds_regdate)) > ?))
THEN 1 ELSE 0 END
) AS prev_end,
SUM(CASE WHEN YEAR(COALESCE(ds_designated_at, DATE(ds_regdate))) = ? THEN 1 ELSE 0 END) AS designated_y,
SUM(
CASE WHEN ds_state IN (2,3)
AND YEAR(COALESCE(ds_state_changed_at, DATE(ds_regdate))) = ?
THEN 1 ELSE 0 END
) AS cancelled_y,
SUM(
CASE WHEN COALESCE(ds_designated_at, DATE(ds_regdate)) <= ?
AND (ds_state = 1 OR (ds_state IN (2,3) AND COALESCE(ds_state_changed_at, DATE(ds_regdate)) > ?))
THEN 1 ELSE 0 END
) AS curr_end
FROM designated_shop
WHERE ds_lg_idx = ?
AND (? = '' OR ds_gugun_code = ?)
GROUP BY gugun_key, zone_key
ORDER BY gugun_key, zone_key
";
} else {
$sql = "
SELECT
IFNULL(NULLIF(TRIM(ds_gugun_code), ''), '') AS gugun_key,
SUM(
CASE WHEN COALESCE(ds_designated_at, DATE(ds_regdate)) <= ?
AND (ds_state = 1 OR (ds_state IN (2,3) AND COALESCE(ds_state_changed_at, DATE(ds_regdate)) > ?))
THEN 1 ELSE 0 END
) AS prev_end,
SUM(CASE WHEN YEAR(COALESCE(ds_designated_at, DATE(ds_regdate))) = ? THEN 1 ELSE 0 END) AS designated_y,
SUM(
CASE WHEN ds_state IN (2,3)
AND YEAR(COALESCE(ds_state_changed_at, DATE(ds_regdate))) = ?
THEN 1 ELSE 0 END
) AS cancelled_y,
SUM(
CASE WHEN COALESCE(ds_designated_at, DATE(ds_regdate)) <= ?
AND (ds_state = 1 OR (ds_state IN (2,3) AND COALESCE(ds_state_changed_at, DATE(ds_regdate)) > ?))
THEN 1 ELSE 0 END
) AS curr_end
FROM designated_shop
WHERE ds_lg_idx = ?
AND (? = '' OR ds_gugun_code = ?)
GROUP BY gugun_key
ORDER BY gugun_key
";
}
$raw = $db->query($sql, $bind)->getResult();
$out = [];
foreach ($raw as $row) {
$gk = (string) ($row->gugun_key ?? '');
$gn = $gk !== '' ? ($gugunMap[$gk] ?? $gk) : '(구·군 미입력)';
if ($granularity === 'dong') {
$zk = (string) ($row->zone_key ?? '');
$zn = $zk !== '' ? $zk : '(구역 미입력)';
$label = $gn . ' / ' . $zn;
$rkey = $gk . "\t" . $zk;
} else {
$label = $gn;
$rkey = $gk;
}
$prev = (int) ($row->prev_end ?? 0);
$curr = (int) ($row->curr_end ?? 0);
$des = (int) ($row->designated_y ?? 0);
$can = (int) ($row->cancelled_y ?? 0);
$zk = $granularity === 'dong' ? (string) ($row->zone_key ?? '') : '';
$out[] = (object) [
'region_key' => $rkey,
'region_label' => $label,
'gugun_code' => $gk,
'zone_code' => $zk,
'prev_end' => $prev,
'designated_y' => $des,
'cancelled_y' => $can,
'curr_end' => $curr,
'delta_curr_prev' => $curr - $prev,
'delta_des_cancel' => $des - $can,
];
}
return $out;
}
/**
* P2-18: 지정판매소 현황 (연도별 신규/취소 + 구·군·동 집계)
*/
public function status()
{
helper('admin');
$lgIdx = admin_effective_lg_idx();
if ($lgIdx === null || $lgIdx <= 0) {
return redirect()->to(work_area_home_url())
->with('error', '작업할 지자체가 선택되지 않았습니다.');
}
$yearRaw = $this->request->getGet('year');
$year = ($yearRaw !== null && $yearRaw !== '') ? (int) $yearRaw : (int) date('Y');
$currentLg = $this->lgModel->find($lgIdx);
$fixedGugunCode = trim((string) ($currentLg->lg_code ?? ''));
$fixedGugunLabel = trim((string) ($currentLg->lg_gugun ?? ''));
if ($fixedGugunLabel === '') {
$fixedGugunLabel = trim((string) ($currentLg->lg_name ?? ''));
}
$db = \Config\Database::connect();
// 연도별 신규등록 건수 (ds_designated_at 기준)
$newByYear = $db->query("
SELECT YEAR(ds_designated_at) as yr, COUNT(*) as cnt
FROM designated_shop
WHERE ds_lg_idx = ? AND ds_designated_at IS NOT NULL
GROUP BY YEAR(ds_designated_at)
ORDER BY yr DESC
", [$lgIdx])->getResult();
// 연도별 취소/비활성 건수 (ds_state != 1, ds_regdate 기준)
$cancelByYear = $db->query("
SELECT YEAR(ds_regdate) as yr, COUNT(*) as cnt
FROM designated_shop
WHERE ds_lg_idx = ? AND ds_state != 1
GROUP BY YEAR(ds_regdate)
ORDER BY yr DESC
", [$lgIdx])->getResult();
// 전체 현황 합계
$totalActive = $this->shopModel->where('ds_lg_idx', $lgIdx)->where('ds_state', 1)->countAllResults(false);
$totalInactive = $this->shopModel->where('ds_lg_idx', $lgIdx)->where('ds_state !=', 1)->countAllResults(false);
$districtRows = $this->designatedShopDistrictStatusRows($lgIdx, $year, $fixedGugunCode, 'gugun');
$zoneRowsRaw = $this->designatedShopDistrictStatusRows($lgIdx, $year, $fixedGugunCode, 'dong');
$zoneSummaryRows = [];
foreach ($zoneRowsRaw as $zr) {
$zoneLabel = trim((string) ($zr->zone_code ?? ''));
if ($zoneLabel === '') {
$zoneLabel = '(구역 미입력)';
}
$zoneSummaryRows[] = (object) [
'zone_label' => $zoneLabel,
'prev_end' => (int) ($zr->prev_end ?? 0),
'designated_y' => (int) ($zr->designated_y ?? 0),
'cancelled_y' => (int) ($zr->cancelled_y ?? 0),
'curr_end' => (int) ($zr->curr_end ?? 0),
'delta_curr_prev' => (int) ($zr->delta_curr_prev ?? 0),
'delta_des_cancel' => (int) ($zr->delta_des_cancel ?? 0),
];
}
usort($zoneSummaryRows, static function ($a, $b): int {
$cmp = $b->curr_end <=> $a->curr_end;
if ($cmp !== 0) {
return $cmp;
}
return strcmp((string) $a->zone_label, (string) $b->zone_label);
});
$sumP = array_sum(array_map(static fn ($r) => $r->prev_end, $districtRows));
$sumD = array_sum(array_map(static fn ($r) => $r->designated_y, $districtRows));
$sumC = array_sum(array_map(static fn ($r) => $r->cancelled_y, $districtRows));
$sumCur = array_sum(array_map(static fn ($r) => $r->curr_end, $districtRows));
$districtTotal = (object) [
'region_label' => '합계',
'gugun_code' => '',
'zone_code' => '',
'prev_end' => $sumP,
'designated_y' => $sumD,
'cancelled_y' => $sumC,
'curr_end' => $sumCur,
'delta_curr_prev' => $sumCur - $sumP,
'delta_des_cancel' => $sumD - $sumC,
];
$yearChoices = [];
$yMax = (int) date('Y');
for ($y = $yMax; $y >= $yMax - 15; $y--) {
$yearChoices[] = $y;
}
return $this->renderWorkPage('지정판매소 현황', 'admin/designated_shop/status', [
'newByYear' => $newByYear,
'cancelByYear' => $cancelByYear,
'totalActive' => $totalActive,
'totalInactive' => $totalInactive,
'districtRows' => $districtRows,
'districtTotal' => $districtTotal,
'zoneSummaryRows'=> $zoneSummaryRows,
'reportYear' => $year,
'fixedGugunLabel'=> $fixedGugunLabel,
'yearChoices' => $yearChoices,
]);
}
/**
* 구·군(·동) 신규/취소 현황 CSV (status 화면과 동일 조건)
*/
public function statusExport()
{
helper(['admin', 'export']);
$lgIdx = admin_effective_lg_idx();
if ($lgIdx === null || $lgIdx <= 0) {
return redirect()->to(work_area_home_url())
->with('error', '작업할 지자체가 선택되지 않았습니다.');
}
$yearRaw = $this->request->getGet('year');
$year = ($yearRaw !== null && $yearRaw !== '') ? (int) $yearRaw : (int) date('Y');
$currentLg = $this->lgModel->find($lgIdx);
$fixedGugunCode = trim((string) ($currentLg->lg_code ?? ''));
$rows = $this->designatedShopDistrictStatusRows($lgIdx, $year, $fixedGugunCode, 'gugun');
$sumP = $sumD = $sumC = $sumCur = 0;
foreach ($rows as $r) {
$sumP += $r->prev_end;
$sumD += $r->designated_y;
$sumC += $r->cancelled_y;
$sumCur += $r->curr_end;
}
$labelCol = '군·구';
$csvRows = [];
$n = 0;
foreach ($rows as $r) {
++$n;
$curr = (int) $r->curr_end;
$prev = (int) $r->prev_end;
$pctShare = $sumCur > 0 ? round(($curr / $sumCur) * 100, 1) : 0.0;
$yoyPct = $prev > 0 ? round((($curr - $prev) / $prev) * 100, 1) : '';
$csvRows[] = [
$n,
$r->region_label,
$r->gugun_code,
$r->prev_end,
$r->designated_y,
$r->cancelled_y,
$r->curr_end,
$r->delta_curr_prev,
$r->delta_des_cancel,
$pctShare,
$yoyPct === '' ? '' : $yoyPct,
];
}
$totYoy = $sumP > 0 ? round((($sumCur - $sumP) / $sumP) * 100, 1) : '';
$csvRows[] = [
'',
'합계',
'',
$sumP,
$sumD,
$sumC,
$sumCur,
$sumCur - $sumP,
$sumD - $sumC,
100,
$totYoy === '' ? '' : $totYoy,
];
export_csv(
'지정판매소_신규취소현황_' . $year . '_' . date('Ymd') . '.csv',
[
'순번',
$labelCol,
'구코드',
'종전(전년도말)',
'지정(' . $year . '년)',
'취소(' . $year . '년)',
'현행(금년도말)',
'증감(현행−종전)',
'지정−취소(' . $year . '년)',
'현행비중(%)',
'전년대비증감률(%)',
],
$csvRows
);
}
/**
* GBMS형 지정 판매소 신규/취소 현황 (구·군은 효과 지자체 마스터 고정, 연도별 조회).
*/
public function districtNewCancel()
{
helper('admin');
$lgIdx = admin_effective_lg_idx();
if ($lgIdx === null || $lgIdx <= 0) {
return redirect()->to(work_area_home_url())
->with('error', '작업할 지자체가 선택되지 않았습니다.');
}
$yearRaw = $this->request->getGet('year');
$year = ($yearRaw !== null && $yearRaw !== '') ? (int) $yearRaw : (int) date('Y');
$year = max(1990, min(2100, $year));
$currentLg = $this->lgModel->find($lgIdx);
$bundle = $this->buildGbmsNewCancelRows($lgIdx, $year);
$yearChoices = [];
$yMax = (int) date('Y');
for ($y = $yMax; $y >= $yMax - 15; $y--) {
$yearChoices[] = $y;
}
return $this->renderWorkPage('지정 판매소 신규/취소 현황', 'admin/designated_shop/district_new_cancel', [
'currentLg' => $currentLg,
'reportYear' => $year,
'yearChoices' => $yearChoices,
'districtRows' => $bundle['rows'],
'districtTotal' => $bundle['total'],
]);
}
/**
* GBMS형 신규/취소 현황 CSV (districtNewCancel 화면과 동일 조건).
*/
public function districtNewCancelExport()
{
helper(['admin', 'export']);
$lgIdx = admin_effective_lg_idx();
if ($lgIdx === null || $lgIdx <= 0) {
return redirect()->to(work_area_home_url())
->with('error', '작업할 지자체가 선택되지 않았습니다.');
}
$yearRaw = $this->request->getGet('year');
$year = ($yearRaw !== null && $yearRaw !== '') ? (int) $yearRaw : (int) date('Y');
$year = max(1990, min(2100, $year));
$bundle = $this->buildGbmsNewCancelRows($lgIdx, $year);
$rows = $bundle['rows'];
$tot = $bundle['total'];
$csvRows = [];
foreach ($rows as $r) {
$csvRows[] = [
$r->region_label,
$r->prev_end,
$r->designated_y,
$r->cancelled_y,
$r->curr_end,
];
}
$csvRows[] = [
$tot->region_label,
$tot->prev_end,
$tot->designated_y,
$tot->cancelled_y,
$tot->curr_end,
];
export_csv(
'지정판매소_신규취소현황_' . $year . '_' . date('Ymd') . '.csv',
[
'군·구',
'종전(전년도말)',
'지정(' . $year . '년)',
'취소(' . $year . '년)',
'현행(금년도말)',
],
$csvRows
);
}
/**
* 기본코드 종류(B·C·D) code_kind PK (미사용·미등록 null).
*/
private function codeKindIdxByCkCode(string $ckCode): ?int
{
$k = model(CodeKindModel::class)
->where('ck_code', $ckCode)
->where('ck_state', 1)
->first();
return $k !== null ? (int) $k->ck_idx : null;
}
/**
* 주소 문자열에 행정구역 명칭이 포함되는지(공백 무시·부분 일치).
*/
private function addressHaystackContainsRegionName(string $haystack, string $name): bool
{
$name = trim($name);
if ($name === '') {
return false;
}
$h = $this->compactAddressText($haystack);
$n = $this->compactAddressText($name);
return $h !== '' && $n !== '' && mb_stripos($h, $n, 0, 'UTF-8') !== false;
}
/**
* 동일 지자체(ds_lg_idx) 소속 판매소번호 , 3자리가 모두 숫자인 것만 모아 최댓값.
* 목록은 3자리만 표시하므로, 구·동 접두(B+C+D) 무관하게 일련이 이어지게 한다.
*/
private function maxDesignatedShopThreeDigitSerialForLocalGovernment(int $lgIdx): int
{
$rows = $this->shopModel->select('ds_shop_no')->where('ds_lg_idx', $lgIdx)->findAll();
$max = 0;
foreach ($rows as $row) {
$no = trim((string) ($row->ds_shop_no ?? ''));
if (strlen($no) < 3) {
continue;
}
$tail = substr($no, -3);
if ($tail === '' || ! ctype_digit($tail)) {
continue;
}
$n = (int) $tail;
if ($n > $max) {
$max = $n;
}
}
return $max;
}
/**
* 판매소번호: 기본코드 B + C + D( cd_code에서 상위 코드 접두 제거 이어 붙임) + 3자리 일련.
* 주소(카카오 시·구·도로명·지번) code_detail만 사용하며 kr_address 외부 참조 테이블은 사용하지 않음.
*
* @return array{ok:true, shop_no:string, gugun_code:string}|array{ok:false, error:string}
*/
private function resolveDesignatedShopNumberFromAddress(
int $lgIdx,
string $addrSido,
string $addrSigungu,
string $road,
string $jibun,
string $zip,
object $lg
): array {
$bCk = $this->codeKindIdxByCkCode('B');
$cCk = $this->codeKindIdxByCkCode('C');
$dCk = $this->codeKindIdxByCkCode('D');
if ($bCk === null || $cCk === null || $dCk === null) {
return [
'ok' => false,
'error' => '기본코드 종류(B·C·D)가 등록되어 있지 않습니다. 시스템 관리자에게 문의하세요.',
];
}
$detailModel = model(CodeDetailModel::class);
$bRows = $detailModel->getByKind($bCk, true, $lgIdx);
$cRows = $detailModel->getByKind($cCk, true, $lgIdx);
$dRows = $detailModel->getByKind($dCk, true, $lgIdx);
$sido = trim($addrSido);
$sig = trim($addrSigungu);
$blob = trim($sido . ' ' . $sig . ' ' . $road . ' ' . $jibun . ' ' . $zip);
if ($blob === '') {
$blob = trim((string) ($lg->lg_sido ?? '') . ' ' . (string) ($lg->lg_gugun ?? ''));
}
$bCode = null;
foreach ($bRows as $row) {
$nm = trim((string) $row->cd_name);
$cd = trim((string) $row->cd_code);
if ($nm === '' || $cd === '') {
continue;
}
if ($this->koreanRegionTokenMatches($nm, $sido, $blob)) {
$bCode = $cd;
break;
}
}
if ($bCode === null || $bCode === '') {
return [
'ok' => false,
'error' => '주소에서 시·도 기본코드(B)를 찾을 수 없습니다. 기본코드(B)에 해당 광역단위를 등록했는지 확인해 주세요.',
];
}
$cCode = null;
foreach ($cRows as $row) {
$cd = trim((string) $row->cd_code);
if ($cd === '' || ! str_starts_with($cd, $bCode)) {
continue;
}
$nm = trim((string) $row->cd_name);
if ($nm === '') {
continue;
}
if ($this->koreanRegionTokenMatches($nm, $sig, $blob)) {
$cCode = $cd;
break;
}
}
if ($cCode === null || $cCode === '') {
return [
'ok' => false,
'error' => '주소에서 구·군 기본코드(C)를 찾을 수 없습니다. 기본코드(C)를 확인하거나 주소 검색 결과(시·군·구)를 확인해 주세요.',
];
}
$dCandidates = [];
foreach ($dRows as $row) {
$cd = trim((string) $row->cd_code);
if ($cd === '' || ! str_starts_with($cd, $cCode)) {
continue;
}
$nm = trim((string) $row->cd_name);
if ($nm === '') {
continue;
}
$dCandidates[] = [
'len' => mb_strlen($nm, 'UTF-8'),
'nm' => $nm,
'cd' => $cd,
];
}
usort($dCandidates, static fn (array $a, array $b): int => $b['len'] <=> $a['len']);
$dCode = null;
foreach ($dCandidates as $cand) {
if ($this->addressHaystackContainsRegionName($blob, $cand['nm'])) {
$dCode = $cand['cd'];
break;
}
}
if ($dCode === null || $dCode === '') {
return [
'ok' => false,
'error' => '주소에서 동 기본코드(D)를 찾을 수 없습니다. 지번·도로명에 법정동명이 포함되는지, 기본코드(D)에 해당 동이 등록되어 있는지 확인해 주세요.',
];
}
$cRest = str_starts_with($cCode, $bCode) ? substr($cCode, strlen($bCode)) : $cCode;
$dRest = str_starts_with($dCode, $cCode) ? substr($dCode, strlen($cCode)) : $dCode;
$prefix = $bCode . $cRest . $dRest;
// 목록 UI는 판매소번호 끝 3자리만 보여 주므로, 동일 지자체(ds_lg_idx) 안에서는
// 구·동 접두와 무관하게 일련(마지막 3자리)이 이어지도록 한다(구형 lg_code+일련과 호환).
$maxSerial = $this->maxDesignatedShopThreeDigitSerialForLocalGovernment($lgIdx);
return [
'ok' => true,
'shop_no' => $prefix . sprintf('%03d', $maxSerial + 1),
'gugun_code' => $cCode,
];
}
/** 카카오맵 JavaScript SDK용 키 (.env kakao.javascriptKey) */
private function kakaoJavascriptKey(): string
{
return (string) (config(\Config\Kakao::class)->javascriptKey ?? '');
}
}