Warning: session_save_path(): Cannot change save path when headers already sent in /home/yqual/s-yqual.com/public_html/s-yqual/public/yqual/wp-content/themes/yqual/session.php on line 2

Warning: session_start(): Cannot start session when headers already sent in /home/yqual/s-yqual.com/public_html/s-yqual/public/yqual/wp-content/themes/yqual/session.php on line 3

Warning: session_regenerate_id(): Cannot regenerate session id - session is not active in /home/yqual/s-yqual.com/public_html/s-yqual/public/yqual/wp-content/themes/yqual/session.php on line 5
プログラミング

フレンドリィショップ編 アイテムの販売 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ポケモン 92
プログラミング
PHP,PHPポケモン,ポケモン
両隣のポケモン取得編(配列の隣の要素)PHPポケモン 92

両隣のポケモンを判別する パーティーの中からID指定でポケモンを特定するためのメソッドは実装されていましたが、その両隣にどんなポケモンが控えているのかを判別する手段がありませんでしたね。 なので、今回は両隣のポケモンを判別するためのメソッドを準備して活用していきましょう。   パーティー まずは...

動画編集で知っておくべき基本的な3工程
動画編集
Adobe,AfterEffects,Illustrator,Photoshop,PremierePro
動画編集で知っておくべき基本的な3工程

  YouTubeに動画を投稿したいけど、動画の基本的な作成手順がわからない・・・   動画作成をするためのソフトは色々ありますが、有料なものであればやはりAdobe製品に軍配が上がります。 しかし準備したは良いものの、いろんなソフトが合ってどれを使えばよいか分からないという悩みを抱えている方は非常...

PHPポケモン「バトルシステム実装編〜補正値計算・乱数・急所〜」21
プログラミング
PHP,PHPポケモン,ポケモン
PHPポケモン「バトルシステム実装編〜補正値計算・乱数・急所〜」21

バトルシステムの実装  今回は「急所」と「乱数」と「タイプ一致」の判定と補正を実装していきます。 ちなみにですが、ポケモンwikiを熟読したところ、補正値の計算にも順番があり、計算後に小数点の切り捨てや五捨五超入をするなど、そこそこ複雑な計算順序がありましたが、今回はそこまで精密に再現せず、補正値(...

【Adobe Animate CC】モーショントゥイーンを使ったアニメーション作成
動画編集
Adobe,Animate,Illustrator,Photoshop,アニメーション
【Adobe Animate CC】モーショントゥイーンを使ったアニメーション作成

    今回はキャラクターがうちわを扇いでいるアニメーションのAdobe Animateを使った作成方法をご紹介します。 完成イメージは以下の通りです。   Photoshopなどを使ってもgifアニメーションの作成はできますが、Adobe Animateのモーショントゥイーンを使えば、変化するポイントだけを設定す...

プレイヤーのグローバル化編 PHPポケモン 78
プログラミング
PHP,PHPポケモン,ポケモン
プレイヤーのグローバル化編 PHPポケモン 78

2日ほどwiki作成にいそいそと励んでいましたが、開発をお休みしていたわけではありません。追加機能を実装するに辺り、色々と改善点が挙がってきたので、このタイミングでしっかりと見直しをしました。   本格的なシステム開発では、最初に仕様書や設計書が作成され、それに沿って作成していくことになります...

Zoomも飲み屋もなくならない 〜コロナ収束後の本当の世界〜
雑記
YouTube,Zoom,コロナ,テレワーク,リモートワーク
Zoomも飲み屋もなくならない 〜コロナ収束後の本当の世界〜

  新型コロナウイルスのニュースが絶えない毎日を過ごす中、コロナの内容に一度も触れていませんでしたがそろそろ触れておこうと思います。   マイナスな内容は不安を煽るような推測はSNSやニュースでイヤというほど目にしているので、可能性を感じられる内容と現実をお届けします。   コロナが収束した時、この...

HPバーアニメーション 後編 フロント側の対応 PHPポケモン 44
プログラミング
PHP,PHPポケモン,ポケモン
HPバーアニメーション 後編 フロント側の対応 PHPポケモン 44

HPバーアニメーション それでは前回に続き、HPバーのアニメーションづくりをしていきましょう。前回、メッセージに合わせてレスポンスを返却するというサーバー側の仕組みを作成しました。なので、今回はそれをフロント側で受け取り、タイミングよくアニメーションで再現します。   フロント側(js)の処理 前回...

ポケモン入れ替え編 PHPポケモン 83
プログラミング
PHP,PHPポケモン,ポケモン
ポケモン入れ替え編 PHPポケモン 83

ポケモンの入れ替え 複数ポケモンの所有、そして並び替えの機能が整ったので、いよいよバトルでのポケモン交代機能を実装します。ポケモンがバトル中に交代する方法は大きく分けて以下の4つです。 プレイヤー操作による交代 ひんしによる交代 相手ポケモンの技による交代 味方ポケモンの技による交代 ...

カテゴリ

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