プログラミング

忘れさせる技選択 後編(新しい技を習得) PHPポケモン 56

PHP PHPポケモン ポケモン
忘れさせる技選択 後編(新しい技を習得) PHPポケモン 56

新しい技を習得

前編に続き、技習得時の処理分岐を作成していきましょう。今回は「新しい技を習得する」です。

前回は覚えようとしている技を諦めるだけだったので、ポケモンのオブジェクトを書き換える必要がありませんでしたね。ですが、新しく覚えようとしている技を既に覚えている技と置き換える場合は、ポケモンオブジェクトの書き換えをする必要があります

 

現在は野生のポケモンを倒して技を覚えるという方法しかないため、書き換えはこのタイミングでは不要だと思うかも知れません。ですが、もしこれが相手トレーナーだった場合はいかがでしょうか?

ポケモンを倒しても、次のポケモンが出てくる可能性があるため、オブジェクトが書き換わっていなければ技選択の選択ができなくなります。また、複数の技を覚えたり、2以上のレベルアップの処理についても想定しておかなければなりません。

 

処理は今までとさほど変わりませんが、データの受け渡しや状況把握が今回複雑になるポイントです。しっかりとオブジェクトや配列の状態を理解しながら進めていきましょう。

 

忘れさせる技の選択

まずは忘れさせる技の選択です。htmlは前回と同様のモーダルを使用します。そこに必要データをjQueryを使って用意してあげましょう。

 

バトル画面用JS/Public/Assets/js/Battle/fight.js
/**
* 技テーブルクリック時の関数
* @function click
* @return void
**/
var selectForgetMoveInit = function(){
    $('.forget-selectmove').on('click', function(){
        // 諦めるボタンの無効化切り替え
        if($(this).hasClass('new-move')){
            $("#btn-abandon-move").prop('disabled', false);
            $('#btn-forget-move').hide();
            return;
        }else{
            $("#btn-abandon-move").prop('disabled', true);
        }
        // 技名を取得
        var name = $(this).data('name');
        // ボタンに技名をセット
        $('#btn-forget-move').find('.move-name')
        .text(name);
        $('#btn-forget-move').show();
    });
}
 
/**
* 技テーブルクリック時の関数
* @function click
* @return void
**/
var submitForgetMoveInit = function(){
    $('#btn-forget-move').on('click', function(){
        // 技をフォームへセット
        var data = {
            id: $(this).data('msg_id'),
            forget: $('.forget-selectmove.active').data('num'),
            level: $('#level').text(),
            hp: $('#hpbar-friend').attr('aria-valuenow')
        };
        // フォームを用意
        $.each(data, function(key, val){
            // フォームの最後にパラメーターを追加
            $('#remote-form').append(
                '<input type="hidden" name="param[' + key + ']" value="' + val + '">'
            );
        })
        // サブミット実行
        $('#remote-form-action').val('learn_move');
        $('#remote-form').submit();
    });
}

 

「技を忘れさせる」というボタンを押すと、テーブルの行(tr)に用意したdataを取得して、リモートフォームのinputとしてセットしています。postする値は、技選択のモーダルを呼び出したメッセージIDと、忘れる技の添え番、現在のレベル、現在のHPの4つです。

 

現画面を再現するための必要値

ただ単にデータをポストして、技を書き換えて画面を開始させた場合、至る場所にズレが生じます。その1つが「レベル」です。6から7にアップするだけなら構いませんが、6から8にレベルアップできる量の経験値を取得して、レベル7で技を習得した場合、画面を戻した時点ではレベルが8になってしまいます。レベル7から8になる流れが飛んでしまうのです

そうならないためにも、現状のレベルを画面再現のためにpostする必要があります。また、レベルアップに合わせてHPが変動する可能性もあります。その際にずれが生じないように現在のHP残量も画面再現用にpostしました。

メッセージIDに関しては、現在どこまで処理が進んでいるかを記録するためにポストしています。

アクションには、新しく「learn_move」という値を指定しました。

 

また、技習得直後の再現には現在のレスポンスデータとメッセージデータの引き継ぎ処理が必要になるためバトル画面で前画面の情報をセッションに用意しておきましょう。

 

バトル画面ヘッド(/Resources/Partials/Layouts/Head/battle.php
// レスポンスデータを引き継ぎ用にセッションへ格納
$_SESSION['__data']['before_reponses'] = $responses;
$_SESSION['__data']['before_messages'] = $controller->getMessages();

 

技の置き換え処理

では技の置き換え処理(learn_move)の分岐とサービスを作成していきましょう。

 

バトルコントローラー(/App/Controllers/Battle/BattleController.php
// ブランチメソッド内分岐
 
/******************************************
* 技の習得
*/
case 'learn_move':
// サービス実行
$service = new LearnMoveService(
    $this->pokemon,
    $_SESSION['__data']['before_reponses'],
    $_SESSION['__data']['before_messages'],
    $this->request('param')
);
$service->execute();
// 描画するポケモン情報を置き換え
$this->before['friend'] = $service->getTmpPokemon();
break;

 

技習得用サービス(/App/Services/Battle/LearnMoveService.php
<?php
$root_path = __DIR__.'/../../..';
// 親クラス
require_once($root_path.'/App/Services/Service.php');
// トレイト
require_once($root_path.'/App/Traits/Service/Battle/ServiceBattleCheckTrait.php');
 
/**
* 技の習得処理
*/
class LearnMoveService extends Service
{
 
    use ServiceBattleCheckTrait;
 
    /**
    * @var Pokemon:object
    */
    protected $pokemon;
 
    /**
    * @var Pokemon:object
    */
    protected $tmp_pokemon;
 
    /**
    * @var array
    */
    protected $before_responses;
 
    /**
    * @var array
    */
    protected $request;
 
    /**
    * @return void
    */
    public function __construct($pokemon, $before_responses, $before_messages, $request)
    {
        $this->pokemon = $pokemon;
        $this->before_responses = $before_responses;
        $this->before_messages = $before_messages;
        $this->request = $request;
    }
 
    /**
    * @return void
    */
    public function execute()
    {
        $this->replaceMove();
        $this->tmp_pokemon = $this->createTmpPokemon();
        // メッセージとレスポンスの引き継ぎ
        $this->setMessage(
            $this->getUntreatedResponses($this->before_messages, $this->request['id'], true)
        );
        $this->setResponse(
            $this->getUntreatedResponses($this->before_responses, $this->request['id'])
        );
    }
 
    /**
    * @return Pokomon:object
    */
    public function getTmpPokemon()
    {
        return $this->tmp_pokemon;
    }
 
    /**
    * 表示用のポケモンオブジェクトを生成
    * @return Pokemon:object
    */
    private function createTmpPokemon()
    {
        $pokemon = clone $this->pokemon;
        // クローンオブジェクトにレベルと残HPをセット
        $pokemon->setLevel($this->request['level']);
        $pokemon->setRemainingHp($this->request['hp']);
        $pokemon->setDefaultExp();
        return $pokemon;
    }
 
    /**
    * 技の置き換え
    * @return void
    */
    private function replaceMove()
    {
        // 忘れる技を取得
        $forget_move = $this->pokemon
        ->getMove($this->request['forget']);
        // 覚えさせる技を取得
        $new_move = new $this->before_responses[$this->request['id']]['move'];
        // 技を覚えさせる
        $this->pokemon
        ->setMove($new_move, $this->request['forget']);
        // メッセージの返却
        $this->setMessage('1 2の ……ポカン!');
        $this->setMessage($this->pokemon->getNickname().'は、'.$forget_move->getName().'の使い方をキレイに忘れた!そして......');
        $this->setMessage($this->pokemon->getNickname().'は新しく、'.$new_move->getName().'を覚えた!');
    }
 
    /**
    * 未処理メッセージ・レスポンスの引き継ぎ処理
    * @return void
    */
    private function getUntreatedResponses($responses, $msg_id, $msg=false)
    {
        if($msg){
            /**
            * メッセージの処理
            */
            $key = array_search(
                $msg_id,
                array_column($responses, 1),
                true
            );
            // 対象メッセージを含め3つ目までを削除した配列を返却
            return array_splice($responses, $key + 3);
        }else{
            /**
            * レスポンスの処理
            */
            // メッセージIDのレスポンスが入った位置を取得
            $key = array_search(
                $msg_id,
                array_keys($responses),
                true
            );
            // メッセージIDより後のレスポンスを返却
            return array_splice($responses, $key + 1);
        }
    }
 
}

 

まずは技の置き換え処理の部分から見ていきましょう。

/**
* 技の置き換え
* @return void
*/
private function replaceMove()
{
    // 忘れる技を取得
    $forget_move = $this->pokemon
    ->getMove($this->request['forget']);
    // 覚えさせる技を取得
    $new_move = new $this->before_responses[$this->request['id']]['move'];
    // 技を覚えさせる
    $this->pokemon
    ->setMove($new_move, $this->request['forget']);
    // メッセージの返却
    $this->setMessage('1 2の ……ポカン!');
    $this->setMessage($this->pokemon->getNickname().'は、'.$forget_move->getName().'の使い方をキレイに忘れた!そして......');
    $this->setMessage($this->pokemon->getNickname().'は新しく、'.$new_move->getName().'を覚えた!');
}

 

覚えさせる技はについては、セッションから受け取った前画面のレスポンスから取り出します。フォームで技クラスを送信してしまうと、ユーザー側が直接書き換えられることで、覚えるはずのない技を習得してしまう可能性があるためです。

技を覚えさせる際に、第2引数で忘れさせる技番号を選択しています。もしこちらが該当しない番号であれば、初期値として0番を渡しているので、一番上にある技が削除対象となります。

 

技の置き換え処理後は、必要なメッセージを順番にセットしていきましょう。

  

ダミーのポケモンオブジェクト

他の処理では、開始時には前の状態のポケモンオブジェクトをスタート時点に描画していました。ですが、今回は処理の途中のため、このまま返却すれば前処理を終えた状態のオブジェクトが描画されてしまいます。そうしないためにも、送信したパラメーターを使って描画用のダミーのポケモンオブジェクトを用意する必要があります。

 

/**
* 表示用のポケモンオブジェクトを生成
* @return Pokemon:object
*/
private function createTmpPokemon()
{
    $pokemon = clone $this->pokemon;
    // クローンオブジェクトにレベルと残HPをセット
    $pokemon->setLevel($this->request['level']);
    $pokemon->setRemainingHp($this->request['hp']);
    $pokemon->setDefaultExp();
    return $pokemon;
}

 

まず、ポケモンオブジェクトをクローンで複製し、そこにパラメーターで受け取ったレベルと残りHPをセットしています。経験値バーもレベルアップ直後は0位置スタートさせるため、現在のレベルの初期経験値を再セットしました。

ここで、ポケモンのオブジェクトを新規作成せずクローンで生成している理由は、個体値と努力値の問題です。

新しく生成すれば、もちろん努力値はすべて0、個体値も異なった「別個体」を使って描画することになります。こうげきやすばやさなどのステータスは良いかも知れませんが、HPは目に見える値のためズレが生じてしまいます。

そうならないためにも、同じ個体の複製を描画用として生成しましょう。

 

setLevelなどprotectedで用意していたメソッドは、サービスからでも呼び出せるようにpublicに変更しておいてください。

生成したクローンは、最終的にコントローラーで置き換えています。

 

レスポンスの引き継ぎ

最後にレスポンスの引き継ぎです。連続レベルアップ処理などを想定した場合、リクエストデータとメッセージデータは引き継ぎしなければなりません。なので、postしたメッセージIDより後のデータだけを配列から取り出して、レスポンスとして返却してあげましょう。

/**
* 未処理メッセージ・レスポンスの引き継ぎ処理
* @return void
*/
private function getUntreatedResponses($responses, $msg_id, $msg=false)
{
    if($msg){
        /**
        * メッセージの処理
        */
        $key = array_search(
            $msg_id,
            array_column($responses, 1),
            true
        );
        // 対象メッセージを含め3つ目までを削除した配列を返却
        return array_splice($responses, $key + 3);
    }else{
        /**
        * レスポンスの処理
        */
        // メッセージIDのレスポンスが入った位置を取得
        $key = array_search(
            $msg_id,
            array_keys($responses),
            true
        );
        // メッセージIDより後のレスポンスを返却
        return array_splice($responses, $key + 1);
    }
}

 

似た処理になるので1メソッドにまとめ引数で分岐させていますが、それぞれメソッドを用意しても構いません。

 

メッセージの場合、メッセージIDは配列内の2つ目(1)の要素として格納されています。なので、array_columnarray_searchを組み合わせてメッセージIDの現在位置以降を判別しています。

配列からの切り出しにはarray_splice関数を使います。 

 

モーダル生成時には「諦めメッセージ」と空メッセージが、セットで格納されているため、取得したキー+3の位置より後のメッセージにずらして返却しています。

 

レスポンスはIDをキーにして格納されているので、array_columnの代わりにarray_keysarray_searchを使って該当位置を特定しています。

 

  • レスポンスとメッセージの前画面からの引き継ぎについてですが、現処理であれば超低確率でメッセージIDが重複する可能性があります。ですが、現状メッセージIDを引き継ぎ後に生成していませんので、こちらは一旦保留としておきます。
  • 現在未実装ですが、モーダルに関しても引き継ぎ処理が必要になります。検証前なのでこちらも一旦保留としています。

 

それでは、新しく覚えようとしている技の習得処理を実装していきましょう。

 

 

既に覚えていた技を忘れ、新しい技を習得することができました。

これで技習得処理は完成です。

 

まとめ

いかがだったでしょうか。

今回・前回のPHPポケモンでは「技習得時の処理」を作成しました。

現在プログラミング学習に取り組んでいる方、ゲームづくりに興味がある人は、ぜひ参考にしてみてくださいね。

 

注目の記事

PHPポケモン「技クラス実装編」14
プログラミング
PHP,PHPポケモン,ポケモン
PHPポケモン「技クラス実装編」14

  前回せっかくBootstrapを使って見た目を整えたにも関わらず、ビューポートの記述が抜けているという凡ミスが発覚したので修正しています。 サーセン。   今回のPHPポケモンでは本格的な技システムを実装していきます。技システムが整えば、皆さん期待のバトルシステムも間近です。セキュリティ面やファイル構成...

起業・独立を考えている人に向けた具体的なアドバイス3選
雑記
起業・独立を考えている人に向けた具体的なアドバイス3選

  「どのタイミングで起業すべき?」 「会社を作るためにはどういったことをすればいいの?」   あなたはこんな悩みを抱えていませんか? 今回は、これから独立・起業をしようと考えている方や、そういった野望をいだいている人に向けて、自らの経験を元に具体的なアドバイスを3つピックアップしてご紹...

定数と静的変数 ピカチュウとイーブイで学ぶオブジェクト指向
プログラミング
PHP,PHPポケモン,イーブイ,オブジェクト指向,ピカチュウ,ポケモン
定数と静的変数 ピカチュウとイーブイで学ぶオブジェクト指向

PHPポケモンも順調に開発が進んでいると思いきや、ふとした気づきが自分の理解力を思い知らせることとなった今日このごろです。 プログラミングは奥が深く、しっかりと段階を追って理解を進めていけば、「これ・・・便利やんけ!」ってなることがかなり多いということがわかります。   それでは、かの有名な黄色い...

【Laravel7】バリデーションメッセージの日本語化【6系対応】
プログラミング
Laravel,Linux,PHP
【Laravel7】バリデーションメッセージの日本語化【6系対応】

  Laravelのバリデーションメッセージは標準だと英語で返ってきてしまいますね。 1つずつ変更する方法もありますが、言語ファイルを作成して一括変更するほうが開発時間の短縮に繋がります。   今回は「Laravel7のバリデーションメッセージを日本語化する方法」をご紹介します。Laravel6系でも同じ方法ででき...

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

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

PHPポケモン「アクション制御編」27
プログラミング
JavaScript,jQuery,PHP,PHPポケモン,ポケモン
PHPポケモン「アクション制御編」27

  今回のPHPポケモンでは主に画面の作り込みをしていきます。 とは言っても、ガッチリCSSを書いてよりゲームらしい見た目にするわけではなく、あくまで「ゲームシステムを再現するため」だけに整えていくのが目的です。   ということで、今回はPHPよりもBootstrapさんとjQueryさんに活躍してもらいます。   ...

できるやつの「雑学力」
雑記
勉強法,雑学力
できるやつの「雑学力」

  テレビのコメンテーター、討論番組の出演者で活躍するほとんどの人は物知りだ。 あなたも「よくそんなことまで知っているな」と思ったことはあるだろう。 現代で起業して成功し続ける人のほとんどが、常に新しい情報を取り入れている。 そして得た情報に対して、歴史情報と照らし合わせ、自分なりの意見や解釈を...

レビューがアフィになる!?SNS式ECサイトとは【ビジネス企画書】
ビジネスモデル
ECショップ,SNS
レビューがアフィになる!?SNS式ECサイトとは【ビジネス企画書】

  多くの人が利用しているSNS 副業やフリーランスに人気なアフィリエイト   それぞれの良い点を上手く利用すれば、完成するビジネスはないか?と考えて思い浮かんだ企画です。 今回は「レビューがアフィになる!?SNS式ECサイトとは」について、新時代のショッピングモール式ネットショップの企画をご紹...

カテゴリ

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