지정판매소 신규/취소 현황을 사용자 지자체 기준으로 고정 조회하도록 정리하고, 동별 요약과 컬럼 설명 툴팁을 추가했습니다. 또한 지정판매소 바코드 출력 메뉴를 전용 URL로 분리하고 선택 인쇄/출력 레이아웃을 GBMS 형태에 맞춰 구현했습니다.
1563 lines
61 KiB
PHP
1563 lines
61 KiB
PHP
<?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 ?? '');
|
|
}
|
|
}
|
|
|