- Published on
Code Completion for Laravel app()
Service Container 
Code Completion အကြောင်း မပြောခင် Laravel Service Container ကို အကြမ်းဖျင်း နည်းနည်း ရှင်းပြချင်ပါတယ်။ Service Container မှာ အဓိက ၂ ပိုင်း ရှိပါတယ်။ Class တွေရဲ့ dependency တွေကို manage လုပ်တာနဲ့ dependency injection လုပ်ပေးတာပါ။
Binding 
Dependency ကို manage လုပ်တယ်ဆိုတာက ဉပမာ ApiClient class တစ်ခုရဲ့ constructor မှာ $endpoint နဲ့ $apiKey လိုတယ်ဆိုရင် config ထဲက ဘယ် key တွေကို pass လိုက်ပါဆိုပြီး သတ်မှတ်တာကို ဆိုလိုတာပါ။
$this->app->bind(ApiClient::class, function () {
    return new ApiClient('https://example.com', 'example-key');
});ဒါဆိုရင် ApiClient class ကို resolve လုပ်ရင် ApiClient ရဲ့ constructor မှာ https://example.com နဲ့ example-key ကို laravel က pass ပေးသွားမှာပါ။ တကယ့် real world project မှာတော့ အပေါ်က code လို ရိုးရိုး မ bind ဘဲ singleton method ကို သုံးပြီး bind ပါတယ်။
$this->app->singleton(ApiClient::class, function () {
    return new ApiClient('https://example.com', 'example-key');
});ဘာကွာလဲဆိုရင် singleton method က တစ်ခါပဲ resolve လုပ်ပေးမှာ။ app(ApiClient::class) ဆိုပြီး object တစ်ခါဆောက်ပြီးရင် နောက်တစ်ခါ နောက်တစ်နေရာမှာ app(ApiClient::class) လို့ခေါ်လည်း object အသစ်မဆောက်ဘဲ ပထမ ဆောက်ထားတဲ့ object ပဲ return ပြန်မှာပါ။
Resolving 
app() ဆိုတာက dependency တွေကို reslove ဖို့သုံးတာပါ။ app(ApiClient::class) လို့ ခေါ်လိုက်ရင် service container မှာ bind ထားတဲ့အတိုင်း ApiClient ရဲ့ constructor မှာ $endpoint နဲ့ $apiKey pass ပြီး create လုပ်ထားတဲ့ object ကိုရမှာပါ။
ဒီလောက်ဆို Service Container အကြောင်းကို အကြမ်းဖျင်း နားလည်မယ်ထင်ပါတယ်။ အသေးစိတ်သိချင်ရင် ဒီမှာ ကြည့်လို့ရပါတယ်။
Code Completion 
Code completion အတွက်ရှင်းပြဖို့ လွယ်သွားအောင် စောနက example ကို အခုလိုပြင်ရေးလိုက်ပါမယ်။
$this->app->singleton('apiClient', function () {
    return new ApiClient('https://example.com', 'example-key');
});app('apiClient') လို့ခေါ်ရင် ApiClient object ရပါမယ်။ အဲ့မှာ ApiClient class ရဲ့ function ဖြစ်တဲ့ login() ကို အခုလို ခေါ်မယ်ဆိုရင် IDE က မသိပါဘူး။
app('apiClient')->login($phone, $otp);ဘာလို့ မသိတာလဲဆိုတော့ Laravel က dependency ကို runtime မှာ reslove လုပ်သွားလို့ပါ။ IDE က php runtime မှာ dependency ဘယ်လို reslove လုပ်သွားတယ်ဆိုတာ မသိနိုင်ပါဘူး။ အဲ့တာကို ဖြေရှင်းဖို့ PhpStorm advanced metadata ကို အသုံးပြုလို့ရပါတယ်။ အဲ့တွက် PhpStorm ရှိဖို့လည်းမလိုပါဘူး။ .phpstorm.meta.php ကို နားလည်တဲ့ PHP code intelligence တစ်ခုရှိရင် ရပါပြီ။
နမူနာအနေနဲ့ vscode မှာ setup လုပ်ပါမယ်။ အရင်ဆုံး .phpstorm.meta.php ကို နားလည်တဲ့ PHP Intelephense extension သွင်းထားရပါမယ်။ ပြီးရင် .phpstorm.meta.php ဆိုတဲ့ file တစ်ခု create ပြီး အခုလိုရေးပါမယ်။
<?php
namespace PHPSTORM_META {
    override(\app(0), map([
        'apiClient' => \App\Services\ApiClient::class,
    ]));
}app() function ကို apiClient string နဲ့လာခေါ်ရင် return type က ApiClient class လို့ သတ်မှတ်လိုက်တာပါ။ ဒါဆိုရင် vscode က အခုလို သိပါပြီ။

IDE က သိစေချင်တဲ့ တခြား dependency injection တွေကိုလည်း map array ထဲမှာ ထပ်ထည့်လို့ရပါတယ်။ အခုလို ကိုယ်တိုင်မလုပ်ချင်ရင် Laravel IDE Helper package သွင်းပြီး php artisan ide-helper:meta ဆိုပြီး run လည်းရပါတယ်။ သူကတော့ ရှိသမျှ class တွေ အားလုံးအတွက် .phpstorm.meta.php file မှာ generate လုပ်သွားမှာပါ။