refactor: 모델 개선 및 관계 정의 수정
- Material 모델 - fillable 속성 추가 (모든 필드 명시) - User 모델 - BelongsToMany 관계 타입 힌트 추가 - Product 모델 - fillable에 unit 필드 추가 - casts 순서 정리 (boolean 그룹화) - ProductComponent 모델 - quantity 캐스트 정밀도 변경 (decimal:4 → decimal:6) - referencedItem() 메서드 추가 (동적 관계 로드) - product(), material() 관계 메서드 수정 (where 조건 추가) - is_default 캐스트 제거 (컬럼 없음) - Tenant 모델 - options 캐스트 추가 (array) - scopeActive() 추가 (trial, active 상태 필터링) - isActive(), isTrial() 헬퍼 메서드 추가 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
@@ -18,6 +18,23 @@ class Material extends Model
|
|||||||
{
|
{
|
||||||
use SoftDeletes, ModelTrait, BelongsToTenant;
|
use SoftDeletes, ModelTrait, BelongsToTenant;
|
||||||
|
|
||||||
|
protected $fillable = [
|
||||||
|
'tenant_id',
|
||||||
|
'category_id',
|
||||||
|
'name',
|
||||||
|
'item_name',
|
||||||
|
'specification',
|
||||||
|
'material_code',
|
||||||
|
'unit',
|
||||||
|
'is_inspection',
|
||||||
|
'search_tag',
|
||||||
|
'remarks',
|
||||||
|
'attributes',
|
||||||
|
'options',
|
||||||
|
'created_by',
|
||||||
|
'updated_by',
|
||||||
|
];
|
||||||
|
|
||||||
protected $casts = [
|
protected $casts = [
|
||||||
'attributes' => 'array',
|
'attributes' => 'array',
|
||||||
'options' => 'array',
|
'options' => 'array',
|
||||||
|
|||||||
@@ -5,6 +5,7 @@
|
|||||||
use App\Models\Commons\File;
|
use App\Models\Commons\File;
|
||||||
use App\Models\Tenants\Tenant;
|
use App\Models\Tenants\Tenant;
|
||||||
use App\Traits\ModelTrait;
|
use App\Traits\ModelTrait;
|
||||||
|
use Illuminate\Database\Eloquent\Relations\BelongsToMany;
|
||||||
use Illuminate\Database\Eloquent\SoftDeletes;
|
use Illuminate\Database\Eloquent\SoftDeletes;
|
||||||
use Illuminate\Foundation\Auth\User as Authenticatable;
|
use Illuminate\Foundation\Auth\User as Authenticatable;
|
||||||
use Illuminate\Notifications\Notifiable;
|
use Illuminate\Notifications\Notifiable;
|
||||||
|
|||||||
@@ -15,19 +15,19 @@ class Product extends Model
|
|||||||
use SoftDeletes, BelongsToTenant, ModelTrait;
|
use SoftDeletes, BelongsToTenant, ModelTrait;
|
||||||
|
|
||||||
protected $fillable = [
|
protected $fillable = [
|
||||||
'tenant_id','code','name','category_id',
|
'tenant_id','code','name','unit','category_id',
|
||||||
'product_type', // 라벨/분류용
|
'product_type', // 라벨/분류용
|
||||||
'attributes','description','is_active',
|
'attributes','description',
|
||||||
'is_sellable','is_purchasable','is_producible',
|
'is_sellable','is_purchasable','is_producible','is_active',
|
||||||
'created_by','updated_by'
|
'created_by','updated_by'
|
||||||
];
|
];
|
||||||
|
|
||||||
protected $casts = [
|
protected $casts = [
|
||||||
'attributes' => 'array',
|
'attributes' => 'array',
|
||||||
'is_active' => 'boolean',
|
|
||||||
'is_sellable' => 'boolean',
|
'is_sellable' => 'boolean',
|
||||||
'is_purchasable' => 'boolean',
|
'is_purchasable' => 'boolean',
|
||||||
'is_producible' => 'boolean',
|
'is_producible' => 'boolean',
|
||||||
|
'is_active' => 'boolean',
|
||||||
];
|
];
|
||||||
|
|
||||||
protected $hidden = [
|
protected $hidden = [
|
||||||
|
|||||||
@@ -28,8 +28,7 @@ class ProductComponent extends Model
|
|||||||
];
|
];
|
||||||
|
|
||||||
protected $casts = [
|
protected $casts = [
|
||||||
'quantity' => 'decimal:4',
|
'quantity' => 'decimal:6',
|
||||||
'is_default' => 'boolean',
|
|
||||||
'created_at' => 'datetime',
|
'created_at' => 'datetime',
|
||||||
'updated_at' => 'datetime',
|
'updated_at' => 'datetime',
|
||||||
'deleted_at' => 'datetime',
|
'deleted_at' => 'datetime',
|
||||||
@@ -39,7 +38,6 @@ class ProductComponent extends Model
|
|||||||
'deleted_at',
|
'deleted_at',
|
||||||
];
|
];
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 상위 제품 (모델/제품)
|
* 상위 제품 (모델/제품)
|
||||||
*/
|
*/
|
||||||
@@ -49,19 +47,35 @@ public function parentProduct()
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 하위 제품 (ref_type = PRODUCT일 때만 의미 있음)
|
* 참조된 제품 또는 자재를 동적으로 가져오기
|
||||||
|
* ref_type에 따라 Product 또는 Material을 반환
|
||||||
*/
|
*/
|
||||||
public function childProduct()
|
public function referencedItem()
|
||||||
{
|
{
|
||||||
return $this->belongsTo(Product::class, 'child_product_id');
|
if ($this->ref_type === 'PRODUCT') {
|
||||||
|
return $this->belongsTo(Product::class, 'ref_id');
|
||||||
|
} elseif ($this->ref_type === 'MATERIAL') {
|
||||||
|
return $this->belongsTo(Material::class, 'ref_id');
|
||||||
|
}
|
||||||
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 하위 자재 (ref_type = MATERIAL일 때만 의미 있음)
|
* 하위 제품 (ref_type = PRODUCT일 때만)
|
||||||
|
*/
|
||||||
|
public function product()
|
||||||
|
{
|
||||||
|
return $this->belongsTo(Product::class, 'ref_id')
|
||||||
|
->where('ref_type', 'PRODUCT');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 하위 자재 (ref_type = MATERIAL일 때만)
|
||||||
*/
|
*/
|
||||||
public function material()
|
public function material()
|
||||||
{
|
{
|
||||||
return $this->belongsTo(Material::class, 'material_id');
|
return $this->belongsTo(Material::class, 'ref_id')
|
||||||
|
->where('ref_type', 'MATERIAL');
|
||||||
}
|
}
|
||||||
|
|
||||||
// ---------------------------------------------------
|
// ---------------------------------------------------
|
||||||
|
|||||||
@@ -50,6 +50,7 @@ class Tenant extends Model
|
|||||||
'expires_at' => 'datetime',
|
'expires_at' => 'datetime',
|
||||||
'last_paid_at' => 'datetime',
|
'last_paid_at' => 'datetime',
|
||||||
'max_users' => 'integer',
|
'max_users' => 'integer',
|
||||||
|
'options' => 'array',
|
||||||
'created_at' => 'datetime',
|
'created_at' => 'datetime',
|
||||||
'updated_at' => 'datetime',
|
'updated_at' => 'datetime',
|
||||||
'deleted_at' => 'datetime',
|
'deleted_at' => 'datetime',
|
||||||
@@ -59,6 +60,30 @@ class Tenant extends Model
|
|||||||
'deleted_at',
|
'deleted_at',
|
||||||
];
|
];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 활성화된 테넌트만 조회하는 스코프
|
||||||
|
*/
|
||||||
|
public function scopeActive($query)
|
||||||
|
{
|
||||||
|
return $query->whereIn('tenant_st_code', ['trial', 'active']);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 테넌트가 활성 상태인지 확인
|
||||||
|
*/
|
||||||
|
public function isActive(): bool
|
||||||
|
{
|
||||||
|
return in_array($this->tenant_st_code, ['trial', 'active']);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 테넌트가 트라이얼 상태인지 확인
|
||||||
|
*/
|
||||||
|
public function isTrial(): bool
|
||||||
|
{
|
||||||
|
return $this->tenant_st_code === 'trial';
|
||||||
|
}
|
||||||
|
|
||||||
// 관계 정의 (예시)
|
// 관계 정의 (예시)
|
||||||
public function plan()
|
public function plan()
|
||||||
{
|
{
|
||||||
|
|||||||
Reference in New Issue
Block a user