プログラミング

フレンドリィショップ編 アイテムの販売 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. アイテム機能の追加(使用不可)

 

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

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

 

注目の記事

今からできる!ブログのアクセスを爆UPさせる3大SNS活用法
マーケティング
Facebook,Instagram,Twitter,ブロガー,ブログ
今からできる!ブログのアクセスを爆UPさせる3大SNS活用法

  ブログを収益化させたいけど、なかなかアクセス数が増えない 記事の質は高いのに、その良さをどうやって伝えれば良いかわからない   SEO対策をする上でも、収益化するためにも記事のクオリティは重要です。 しかし、せっかく良い記事を書いていても、そのブログや記事の存在を伝えることができなけれ...

戦闘用アイテム編 プラスパワー PHPポケモン96
プログラミング
PHP,PHPポケモン,ポケモン
戦闘用アイテム編 プラスパワー PHPポケモン96

戦闘用アイテムとは バトル中に使用できるアイテムはいくつかありますが、その中でも「バトル専用」のアイテムがあります。それが戦闘用アイテムであり、主にドーピングと呼ばれるものです。 アイテムカテゴリとして、プレイヤー対象(スプレーなど)、敵ポケモン対象(ボール類)、味方対象(キズぐすり)の3つに...

PHPポケモン「バトルシステム編〜状態異常2〜」31
プログラミング
PHP,PHPポケモン,ポケモン
PHPポケモン「バトルシステム編〜状態異常2〜」31

  前回に引き続き、状態異常チェックを実装します。 まず、前回実装した「ねむり」の処理についてですが、やはりターン数をセットして経過ターン数を引いていくという処理の方が解除率もゲーム再現になるので、まず修正をしておきます。サーセン。   チェック格納トレイト(/Traits/Battle/CheckTrait.php) ...

フリーランスなら心がけておきたい3つのルール【仕事と遊びは両立させろ】
フリーランス
フリーランス,独立
フリーランスなら心がけておきたい3つのルール【仕事と遊びは両立させろ】

  フリーランスになっても不安がいっぱい   会社というものに縛られないというのは楽なイメージがありますが、それ相応の不安がついてまわります。 その結果、会社員へと舞い戻ってしまうと再度そこから抜け出すことは非常に困難です。   今回は、現在活動している人や、これから独立しようとしている...

ポケモンセンター編 PHPポケモン 37
プログラミング
PHP,PHPポケモン,ポケモン
ポケモンセンター編 PHPポケモン 37

ポケモンセンター  バトルシステムを作る関係上、どうしてもダメージを受けることが多くなってきたので、そろそろポケモンセンターを建設(作成)します。 とは言っても、ポケモンセンターに行って交換やらボックス整理などができるわけではなく、ただ回復ポイントを設置するだけの簡単な処理です。   HPの回復 ...

PHPポケモン(α)攻略wiki「稼ぎ方特集」
雑記
PHP,PHPポケモン,ポケモン
PHPポケモン(α)攻略wiki「稼ぎ方特集」

前回に引き続き、連続wiki投稿でPHPポケモン溺愛ユーザーにとっては歓喜の2日間です。   という冗談も踏まえつつ、今回は「稼ぐ」ということについて真面目に考察してみたいと思います。もしリアル世界での「稼ぐ」を目的に来た人は、盛大にブラウザバックしてください。   PHPポケモンにおける「稼ぐ」...

放物線アニメーション編 PHPポケモン 81
プログラミング
PHP,PHPポケモン,ポケモン
放物線アニメーション編 PHPポケモン 81

ボールアニメーション 前回までに作成した捕獲判定処理を使って、ボールのアニメーションを作成します。 捕獲演出は以下の通りです。  味方側から相手に向かってボールを投げる 相手ポケモンの前でボールを開く 捕獲判定で算出した揺れ回数分ボールを揺らす   捕まえた際は、ボールの揺れをストップ...

毎日継続をするためのコツ
雑記
毎日継続をするためのコツ

定期的にコラムを書きたくなるので、今回は「毎日継続をするためのコツ」というテーマで自分が意識していることや、感じたことを書き綴っていきます。 決して開発や業務で追われていたり、PHPポケモンの大幅見直しを迫られているわけではありません。   毎日継続するために  ブログや学習など、毎日継続...

カテゴリ

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 独立 神戸 福祉 秘密鍵 翻訳 自己啓発 英語 見積書 計算機 読書 起業 迷惑メール 配列 銀の弾丸 集客 雑学力