プログラミング

フレンドリィショップ編 アイテムの販売 PHPポケモン 76

PHP PHPポケモン ポケモン
フレンドリィショップ編 アイテムの販売 PHPポケモン 76

リュックの作成

前回はフレンドリィショップへ商品を並べ、計算機を作成するところまで作成しました。ですが、商品が購入できたとしても、それを保管しておくためのスペースがなければ意味がありません

なので、プレイヤー情報に対してアイテムを格納できるように機能拡張をしましょう。

 

プレイヤークラス(/Classes/Player.php
<?php
$root_path = __DIR__.'/..';
// トレイト
require_once($root_path.'/App/Traits/Class/Player/ClassPlayerItemTrait.php');
require_once($root_path.'/App/Traits/Class/Player/ClassPlayerBadgeTrait.php');
require_once($root_path.'/App/Traits/Class/Player/ClassPlayerMoneyTrait.php');
/**
* プレイヤー情報
*/
class Player
{
    use ClassPlayerItemTrait;
    use ClassPlayerBadgeTrait;
    use ClassPlayerMoneyTrait;
 
--省略
 
    /**
    * 持ち物
    * [[class => string, count => int|null], ...]
    * @var array
    */
    protected $items = [];

 ※メソッドが多くなってきたので、それぞれをトレイト分けすることで管理しました

 

アイテムは、配列として格納します。構成は、アイテムの種類1つに対して配列としてクラス名(class)と所有数(count)を持たせました。オブジェクトとして格納してしまうと、サニタイズの処理に負荷がかかるのと、クラス変更による反映が難しくなるため、上記の構成にしています。

  

カテゴリ分けして整頓

次に、カテゴリ分けについてです。リュック内で見やすくするためにも、アイテムに設定した「ボール」や「回復」などのように分けて表示させなければいけません。ですが、アイテムプロパティの配列内を、さらにカテゴリで多次元化すれば、管理する上では複雑化してしまいます。なので、カバンとして取得する際にカテゴリ分け・インスタンス化する機能を持たせたメソッドを用意します。

 

アイテム管理用トレイト(/App/Traits/Class/Player/ClassPlayerItemTrait.php)
<?php
trait ClassPlayerItemTrait
{
    /**==================================================================
    * どうぐ
    ==================================================================**/
 
    /**
    * かばんの生成
    * @return array
    */
    public function getBag(): array
    {
        // carry初期値
        $initial = array_map(function(){
            return [];
        }, array_flip(config('item.categories')));
        // カテゴリ分けした配列を返却
        return array_reduce($this->items, function($carry, $row){
            $item = new $row['class'];
            $carry[$item->getCategory()][] = [
                'item' => $item,
                'count' => $row['count'],
            ];
            return $carry;
        }, $initial);
    }
}

 

配列をカテゴリ分けするために、array_reduceという関数を使用します。 

 

第1引数にカテゴリ分けしたい配列、第2引数にコールバック関数、第3引数にカテゴライズの初期配列を設定することができます。第3引数は、コールバック関数の第1引数の初期値となり、コールバック関数の第2引数ではループさせている配列の値(第1引数)が格納されます。

コールバック関数が含まれるものは、実際に使ってみなければその挙動はわかりにくいため、処理順に見ていきましょう。

// carry初期値
$initial = array_map(function(){
    return [];
}, array_flip(config('item.categories')));

 

まずはカテゴリ分けするための初期値として、カテゴリ名がキー、要素が空の配列を用意します。config(‘item.categories’)の取得結果は以下のようになっています。

[
    'general', 'health', 'ball', 'machine', 'important'
]

 

array_mapの入力配列として使用する際に、array_flipconfigから受け取ったカテゴリ名のキーと要素を逆転させています。

 

array_flip後の配列状態は以下の通りです。

[
    'general' => 0 , 'health' => 1, 'ball' => 2, 'machine' => 3, 'important' => 4
]

 

キー(添字)と値が入れ替わっていることがわかりますね。出来上がった配列をarray_mapを使って、値をから配列に変換した結果が以下の通りです。

[
    'general' => [] , 'health' => [], 'ball' => [], 'machine' => [], 'important' => []
]

 

出来上がった配列に対して、該当するアイテムをarray_reduceを使ってそれぞれ割り振っていきます。

// カテゴリ分けした配列を返却
return array_reduce($this->items, function($carry, $row){
    $item = new $row['class'];
    $carry[$item->getCategory()][] = [
        'item' => $item,
        'count' => $row['count'],
    ];
    return $carry;
}, $initial);

 

$carryには先程作成したカテゴリ配列($initial)、$rowにはアイテム([class, count])が入っています。まず初めにアイテムをインスタンス化、そこからクラス名を取得して$carryの該当配列へ、アイテムのインスタンスと個数を配列として格納しています。

これで、array_reduceの返り値がカテゴライズされたアイテム配列になるので、そのままループして出力すればカテゴライズされた状態が簡単に作り出せます。

ただカテゴライズするだけであれば、$carryの初期値は設定する必要がありません。ですが、リュックには対象カテゴリのアイテムが存在していなくてもカテゴリ名は表示したかったので、初期値を用意してからループさせました。

 

あとは、ポケモンやプレイヤー同様にアイテム用のモーダルを作成して、上記のメソッドを使って出力していきます。

 

アイテムモーダル(/Resources/Partials/Home/Modales/item.php
<!-- Modal -->
<?php $bag = $player->getBag(); ?>
<div class="modal fade" id="item-modal" tabindex="-1" role="dialog" aria-labelledby="item-modal-title" aria-hidden="true">
    <div class="modal-dialog modal-dialog-centered" role="document">
        <div class="modal-content">
            <div class="modal-header">
                <h5 class="modal-title" id="item-modal-title">どうぐ</h5>
                <button type="button" class="close" data-dismiss="modal" aria-label="Close">
                    <span aria-hidden="true">×</span>
                </button>
            </div>
            <div class="modal-body my-2">
                <?php # タブ ?>
                <nav class="nav nav-pills nav-justified btn-group mb-3" id="item-modal-tab">
                    <?php $cnt = 0; ?>
                    <?php foreach($bag as $category => $items): ?>
                        <a class="btn btn-outline-secondary nav-item nav-link <?php if(!$cnt) echo 'active'; ?>" id="item-modal-<?=$category?>-tab" data-toggle="tab" href="#item-modal-<?=$category?>" role="tab" aria-controls="item-modal-<?=$category?>" aria-selected="true">
                            <img src="/Assets/img/item/category/<?=$category?>.png" alt="<?=transJp($category, 'item');?>">
                        </a>
                        <?php $cnt++; ?>
                    <?php endforeach; ?>
                </nav>
                <?php # コンテンツ ?>
                <div class="tab-content" id="item-modal-tab-content">
                    <?php $cnt = 0; ?>
                    <?php foreach($bag as $category => $items): ?>
                        <div class="tab-pane fade show <?php if(!$cnt) echo 'active'; ?>" id="item-modal-<?=$category?>" role="tabpanel" aria-labelledby="item-modal-<?=$category?>">
                            <h6 class="font-weight-bolder mb-2"><?=transJp($category, 'item');?></h6>
                            <div class="bg-light p-3 mb-2 overflow-auto" style="height:120px;">
                                <h6 id="item-modal-<?=$category?>-name" class="font-weight-bolder"></h6>
                                <hr>
                                <p class="mb-0 small" id="item-modal-<?=$category?>-description"></p>
                            </div>
                            <div class="bg-light p-3 overflow-auto" style="height:300px;">
                                <?php if(empty($items)): ?>
                                    <p class="mb-0">1つも持っていません</p>
                                <?php else: ?>
                                    <table class="table table-sm table-hover table-selected table-bordered bg-white mb-0">
                                        <tbody>
                                            <?php foreach($items as $item): ?>
                                                <tr data-description="<?=$item['item']->getDescription()?>"
                                                    data-name="<?=$item['item']->getName()?>"
                                                    data-category="<?=$category?>"
                                                    class="item-row">
                                                    <td class="w-75">
                                                        <img src="/Assets/img/item/class/<?=get_class($item['item'])?>.png" alt="<?=$item['item']->getName()?>" class="mr-1" />
                                                        <?=$item['item']->getName()?>
                                                    </td>
                                                    <td class="w-25 text-right"><?=$item['count']?> 個</td>
                                                </tr>
                                            <?php endforeach; ?>
                                        </tbody>
                                    </table>
                                <?php endif; ?>
                            </div>
                        </div>
                        <?php $cnt++; ?>
                    <?php endforeach; ?>
                </div>
            </div>
        </div>
    </div>
</div>

 

 アイテムを選択すると、上部の小窓に説明分を表示させるようにしました。アイテムを使う処理は未実装のため、現状ではあくまでアイテムが一覧として確認できるだけになります。

 

アイテムの購入

リュックの準備が整ったので、次はアイテムの購入処理に移りましょう。前回入力画面については作成したので、サービス側の処理とデータの受け渡し部分を作り込んでいきます。

 

リクエストのグローバル化

今までPOSTデータはコントローラーのトレイトとしてサニタイズ・ポストデータの取得処理を組み込んでいましたが、こちらもコントローラー・サービス問わずにどこからでも呼び出せるよう、この機会にグローバル化させます。

 

リクエストクラス(/Classes/Request.php
<?php
// リクエスト(送信データの格納オブジェクト)
class Request
{
 
    /**
    * @var array
    */
    private $post = [];
 
    /**
    * @return void
    */
    public function __construct()
    {
        $this->post = $this->sanitize($_POST);
    }
 
    /**
    * @return array
    */
    public function sanitize($array)
    {
        $post = [];
        foreach($array ?? [] as $key => $data){
            if(preg_match('/^__/', $key)){
                // 接頭語にアンダーバーが2つついていればサニタイズ不要
                continue;
            }
            if(is_array($data)){
                // 配列ならループ
                $post[htmlspecialchars($key)] = $this->sanitize($data);
            }else{
                $post[htmlspecialchars($key)] = htmlspecialchars($data);
            }
 
        }
        return $post;
    }
 
    /**
    * 送信データの取得(ドット記法対応)
    * @param dot_key:string
    * @return mixed
    */
    public function request($dot_key)
    {
        $keys = explode('.', $dot_key);
        $values = $this->post;
        foreach($keys ?? [] as $key){
            $values = $values[$key] ?? '';
        }
        return $values;
    }
 
}

 

こちらもドット記法で多次元配列を取得できるように、requestの処理内でexplodeからのforeachで要素を取り出す仕様にしました。

requestのグローバル化は以下のとおりです。

 

リクエストのグローバル関数(/App/Globals/RequestGlobal.php
<?php
// クラス読み込み
require_once($root_path.'/Classes/Request.php');
$global_request = new Request;
/**
* 送信された値の取得
* @param dot_key:string
* @return mixed
*/
function request($dot_key='')
{
    global $global_request;
    return $global_request->request($dot_key);
}

 

これでサービスへの引数を経由させることなく呼び出せるようになりました。

 

お金の計算と商品の提供

それでは、商品の購入・おこづかいの計算処理を、作成したグローバルリクエストを用いながらサービス化していきましょう。

 

フレンドリィショップ用サービス(/App/Services/Home/ShopService.php
<?php
$root_path = __DIR__.'/../../..';
// 親クラス
require_once($root_path.'/App/Services/Service.php');
 
/**
* フレンドリィショップ
*/
class ShopService extends Service
{
 
    /**
    * @var object::Player
    */
    protected $player;
 
    /**
    * @return void
    */
    public function __construct($player)
    {
        $this->player = $player;
    }
 
    /**
    * @return void
    */
    public function execute()
    {
        switch (request('do')) {
            // 購入
            case 'buy':
            $this->buy();
            break;
            // 売却
            case 'sell':
            $this->sell();
            break;
        }
    }
 
    /**
    * 購入
    * @return void
    */
    private function buy(): void
    {
        $class = config('shop.'.request('order'));
        if(empty($class)){
            setMessage('指定されたアイテムは販売しておりません');
            return;
        }
        // アイテムをインスタンス化
        $item = new $class;
        // 購入金額の算出
        $price = $item->getBidPrice() * request('count');
        if($this->player->getMoney() < $price){
            setMessage('おこづかいが足りません');
            return;
        }
        // 残金調整とアイテムの獲得処理
        $result = $this->player
        ->addItem($item, request('count'));
        if($result){
            $this->player
            ->subMoney($price);
            setMessage('毎度ありがとうございました');
        }else{
            setMessage('お客さん、そんなに持てませんよ');
        }
    }
 
    /**
    * 売却
    * @return void
    */
    private function sell(): void
    {
        //
    }
 
}

 

売却機能は一旦後回しにして、購入機能のみを実装しました。

販売していないアイテムを購入できないよう、注文番号で値を受け取り、config内と照合しています。また、フロント側でも制御していますが、お小遣いを超過していれば購入できないようにバックグラウンド側でも制限をかけています

アイテムの追加処理は、プレイヤークラスにメソッドとして持たせています。

 

アイテムの追加(/App/Traits/Class/Player/ClassPlayerItemTrait.php
/**
* アイテムの追加
* @param item:object::Item
* @param count:integer
* @return boolean
*/
public function addItem(object $item, $count=1): bool
{
    if(is_null($item->getMax())){
        // 個数計算しないアイテムの場合はnullをセット
        $count = null;
    }else{
        // 数チェック
        if($count < 1){
            $count = 1; # 最小値
        }
        if($item->getMax() < $count){
            $count = $item->getMax(); # 最大値
        }
    }
    // アイテムのクラスを取得
    $class = get_class($item);
    // 現在所有しているかどうかの確認
    $key = array_search(
        $class,
        array_column($this->items, 'class')
    );
    if($key === false){
        // 所有していない
        $this->items[] = [
            'class' => $class,
            'count' => $count
        ];
        return true;
    }else{
        // 個数計算するアイテム且つ最大量を超過しなければ個数を加算
        if(
            !is_null($count) &&
            ($this->items[$key]['count'] + $count) <= $item->getMax()
        ){
            $this->items[$key]['count'] += $count;
        }else{
            return false;
        }
        return true;
    }
}

 

現在未実装ですが、自転車などのアイテムは複数個所有することができません。なので、そういったアイテムも考慮しつつ、追加処理を作成しました。

既に所有している場合はcountを追加、持っていなければ新しく配列として追加するようにしています。

 

では実際にアイテム購入の流れを見てみましょう。

 

アイテムを購入することができましたね。これでフレンドリィショップの購入処理は完成です。

  

本番環境への反映

PHPポケモン第76回時点での最新版を本番環境へ反映させました。エラー回避のため、セーブデータのリセットを実行しています。

前回からの追加要素は以下の通りです。 

  1. プレイヤー機能の実装
  2. 野生ポケモンの追加(ニャース・メタモン)
  3. フレンドリィショップの追加(購入のみ)
  4. アイテム機能の追加(使用不可)

 

その他細かな要素も追加されていますので、是非遊んでみてください。

最低限の動作チェックは行なっておりますが、α版のため不具合が発生する場合があります。もしお気づきの点がありましたら、お問い合わせいただけると幸いです。

 

注目の記事

進化の石編(実装)PHPポケモン103
プログラミング
PHP,PHPポケモン,ポケモン
進化の石編(実装)PHPポケモン103

アイテムによる進化 前回作成した、進化アイテムによる構成の続きです。   進化アイテムとして「かみなりのいし」、ピカチュウの進化判定を作成したので、アイテムの使用から進化処理までを作成、実装します。   今回作成する処理は以下の2点です。 アイテムの使用判定 進化画面への移管   それ...

目先の利益に気をつけろ!貧乏ビジネスという落とし穴
フリーランス
目先の利益に気をつけろ!貧乏ビジネスという落とし穴

  目先の利益を求めてしまい、来たるべきビジネスチャンスに対応できないというケースは貧乏ビジネスに陥る大きな要因になります。また、相手が下す評価に左右されてしまうことも、自らの評価を下げてしまったり、見積もりを作る上でも大きく影響を及ぼしてしまいます。   今回は「目先の利益に気をつけろ!貧...

PHPポケモン「2進化ポケモン実装編」8
プログラミング
PHP,PHPポケモン,プログラミング学習,ポケモン
PHPポケモン「2進化ポケモン実装編」8

  記念すべき?第8回目で遂にタイトル変更です。 (旧)ピカチュウから学ぶオブジェクト指向 (新) PHPポケモン   機能増設によりオブジェクト指向云々より、PHP学習がメインになったので泣く泣く変更です。(今の所)順調に続いているのも、多くの方?が見てくれているおかげです。感謝感激雨ア...

フリーランスが見積書を作るときに押さえておきたい3つのポイント+α
フリーランス
フリーランス,仕事依頼,独立,見積書
フリーランスが見積書を作るときに押さえておきたい3つのポイント+α

  仕事の依頼がきたけど、どれぐらいの金額を提示すればいいかわからない   駆け出しのフリーランスや、これから独り立ちしようとしている人に多い悩みです。 今回はそういった方のために「フリーランスが見積書を作るときに押さえておきたい3つのポイント+α」についてご紹介します。     時給...

アウトプットのための3つの習慣【3対7を成立させよう】
雑記
アウトプットのための3つの習慣【3対7を成立させよう】

  インプットとアウトプットの比率は3対7がベストだと言われています。 しかし、簡単にできるインプットに比べて、アウトプットは習慣化させておくことが大切です。それができていない人の多くが、膨大に本を読んだり学習に取り組んでも身につかず、趣味のレベルで終わってしまうのです。   今回は、そんな...

独立するならWordPress理解しておけばOK!プログラミングでフリーランスはこれ一つで成り立ちます
プログラミング
PHP,WordPress,フリーランス,独立
独立するならWordPress理解しておけばOK!プログラミングでフリーランスはこれ一つで成り立ちます

  プログラミングでフリーランスを目指すには、どの言語始めればいいの?   プログラミングの学習を始めたのに、それをどう活かせばよいか分からず、いざフリーランスで活動しようと思ってもイメージできずに断念してしまう人は多いです。 言語にも向き不向きがあるため、フリーランスとして活動するために向...

【Laravel】論理削除対応型existsバリデーションの実装方法
プログラミング
Laravel,PHP
【Laravel】論理削除対応型existsバリデーションの実装方法

  Laravelでは多くのバリデーションが提供されていますが、論理削除を使用している場合はそのままでは使えないものが複数あります。 今回は紐付けをする際に存在チェックで使用するexistsのソフトデリート対応のバリデーションを実装する方法をご紹介します。     カスタムバリデーションの追加   存...

Laravelで生成したCookie情報をjQueryで取得する方法【JavaScript】
プログラミング
ajax,api,JavaScript,jQuery,Laravel,PHP
Laravelで生成したCookie情報をjQueryで取得する方法【JavaScript】

  今回はLaravel開発備忘録です。 ajaxでapi認証してviewに記述したhtmlデータを取得するために、cookieを使ったapi_tokenの受け渡し手順をまとめてみました。   Laravelを使った開発をしている人は、ぜひ参考にしてくださいね。     Laravel側の処理   まずはcookieにデータをセットする必要があり...

カテゴリ

SEO対策 イベント デザイン ネットワーク ビジネスモデル フリーランス プログラミング マーケティング ライティング 動画編集 雑記

タグ

5G Adobe AfterEffects AI ajax amazon Animate api artisan atom Automator AWS Bluetooth CSS CVR description EC-CUBE4 ECショップ ESLint Facebook feedly foreach function Google Google AdSense Honeycode htaccess HTML IEEE 802.11ax Illustrator Instagram IoT JavaScript jQuery jQuery UI keyword LAN Laravel Linux MacBook MAMP meta MLM MySQL NoCode note OS OSI参照モデル Paypal Photoshop PHP phpMyAdmin PHPポケモン PremierePro rss SEO SEO対策 Sequel Pro Skype SNS SSH Symfony TCP/IP title Toastr Trait Twig Twitter UCC V系 WAN WebSub Wi-Fi wiki Windows WordPress XAMPP xml Xserver YouTube YouTuber Zoom アーティスト アウトプット アクセス層 アニメーション アフィリエイト イーブイ インターネット インプット エンジニア オブジェクト指向 お金配り クリック単価 クリック数 コミュニケーション能力 コロナ コンサルティング サムネイル システムエンジニア スタートアップ スタイルシート スパム データベース ディープフェイク デザイナー デザイン テレワーク ナンパ ニュース ネットワークモデル ノマドワーク バナー ピカチュウ ビジネス フィード フリーランス ブロガー ブログ プログラマー プログラミング プログラミング学習 プログラミング教育 プロトコル ホームページ制作 ポケモン マークアップ マーケティング メール リモートワーク レンダリング 三井住友 三宮 仕事依頼 児童デイ 児童デイサービス 児童発達支援 公開鍵 初心者 助成金 勉強法 営業 広告 広告収入 必勝マニュアル 放課後等デイサービス 朝活 楽天 深層学習 無線LAN 独立 神戸 福祉 秘密鍵 翻訳 自己啓発 英語 見積書 計算機 読書 起業 迷惑メール 配列 銀の弾丸 集客 雑学力