From c4227f0bbc954d63576acfcbd2596401f2461a33 Mon Sep 17 00:00:00 2001 From: Kevin Whitaker Date: Mon, 20 Apr 2015 20:04:58 -0400 Subject: [PATCH] Add in player management and improve use interaction cases. --- .../src/main/AndroidManifest.xml | 3 + .../CampaignDetailsActivity.java | 40 ++++++-- .../miceandmystics/PlayerManagerActivity.java | 43 ++++++++ .../adapters/PlayerManagerAdapter.java | 92 ++++++++++++++++++ .../viewholders/PlayerViewHolder.java | 31 ++++++ .../src/main/res/layout/item_player.xml | 27 +++++ .../main/res/menu/menu_campaign_details.xml | 8 ++ .../src/main/res/menu/menu_player_manager.xml | 8 ++ .../res/mipmap-xxhdpi/ic_menu_friendslist.png | Bin 0 -> 3228 bytes .../main/res/mipmap-xxhdpi/ic_menu_star.png | Bin 0 -> 3534 bytes 10 files changed, 244 insertions(+), 8 deletions(-) create mode 100644 MiceAndMysticsTracker/src/main/java/com/eyecreate/miceandmystics/miceandmystics/PlayerManagerActivity.java create mode 100644 MiceAndMysticsTracker/src/main/java/com/eyecreate/miceandmystics/miceandmystics/adapters/PlayerManagerAdapter.java create mode 100644 MiceAndMysticsTracker/src/main/java/com/eyecreate/miceandmystics/miceandmystics/viewholders/PlayerViewHolder.java create mode 100644 MiceAndMysticsTracker/src/main/res/layout/item_player.xml create mode 100644 MiceAndMysticsTracker/src/main/res/menu/menu_player_manager.xml create mode 100644 MiceAndMysticsTracker/src/main/res/mipmap-xxhdpi/ic_menu_friendslist.png create mode 100644 MiceAndMysticsTracker/src/main/res/mipmap-xxhdpi/ic_menu_star.png diff --git a/MiceAndMysticsTracker/src/main/AndroidManifest.xml b/MiceAndMysticsTracker/src/main/AndroidManifest.xml index 20ca41e..7f29f01 100644 --- a/MiceAndMysticsTracker/src/main/AndroidManifest.xml +++ b/MiceAndMysticsTracker/src/main/AndroidManifest.xml @@ -25,6 +25,9 @@ android:name="android.support.PARENT_ACTIVITY" android:value="com.eyecreate.miceandmystics.miceandmystics.CampaignActivity" /> + + diff --git a/MiceAndMysticsTracker/src/main/java/com/eyecreate/miceandmystics/miceandmystics/CampaignDetailsActivity.java b/MiceAndMysticsTracker/src/main/java/com/eyecreate/miceandmystics/miceandmystics/CampaignDetailsActivity.java index 3085bc4..47d4dde 100644 --- a/MiceAndMysticsTracker/src/main/java/com/eyecreate/miceandmystics/miceandmystics/CampaignDetailsActivity.java +++ b/MiceAndMysticsTracker/src/main/java/com/eyecreate/miceandmystics/miceandmystics/CampaignDetailsActivity.java @@ -3,7 +3,9 @@ package com.eyecreate.miceandmystics.miceandmystics; import android.app.AlertDialog; import android.content.Context; import android.content.DialogInterface; +import android.content.Intent; import android.os.Bundle; +import android.os.PersistableBundle; import android.support.v7.widget.LinearLayoutManager; import android.view.*; import android.widget.ArrayAdapter; @@ -25,11 +27,20 @@ public class CampaignDetailsActivity extends RecyclerViewActivity { protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setTitle("Game Details"); - campaign = MiceAndMysticsApplication.getRealmInstance().where(Campaign.class).equalTo("campaignName",getIntent().getStringExtra("campaignName")).findFirst(); + if(getIntent().hasExtra("campaignName")) { + campaign = MiceAndMysticsApplication.getRealmInstance().where(Campaign.class).equalTo("campaignName",getIntent().getStringExtra("campaignName")).findFirst(); + } else if(savedInstanceState.containsKey("campaignName")) { + campaign = MiceAndMysticsApplication.getRealmInstance().where(Campaign.class).equalTo("campaignName",savedInstanceState.getString("campaignName")).findFirst(); + } setLayoutManager(new LinearLayoutManager(this)); setAdapter(new CampaignDetailsAdapter(campaign)); } + @Override + public void onSaveInstanceState(Bundle outState, PersistableBundle outPersistentState) { + super.onSaveInstanceState(outState, outPersistentState); + outState.putString("campaignName", campaign.getCampaignName()); + } @Override public boolean onCreateOptionsMenu(Menu menu) { @@ -50,31 +61,38 @@ public class CampaignDetailsActivity extends RecyclerViewActivity { newCharacterDialog(); return true; } else if (id == R.id.action_add_player) { - newPlayerDialog(); + newPlayerDialog(this); + return true; + } else if (id == R.id.action_manage_players) { + Intent managePlayers = new Intent(this,PlayerManagerActivity.class); + startActivity(managePlayers); return true; } return super.onOptionsItemSelected(item); } - public void newPlayerDialog() { - LayoutInflater inflator = (LayoutInflater)(new ContextThemeWrapper(this, R.style.dialogTheme)).getSystemService(Context.LAYOUT_INFLATER_SERVICE); + public static void newPlayerDialog(final Context ctx) { + LayoutInflater inflator = (LayoutInflater)(new ContextThemeWrapper(ctx, R.style.dialogTheme)).getSystemService(Context.LAYOUT_INFLATER_SERVICE); final View dialogView = inflator.inflate(R.layout.dialog_new_player, null, false); final EditText playerEdit = ((EditText)dialogView.findViewById(R.id.player_name)); - AlertDialog addDialog = new AlertDialog.Builder(this,R.style.dialogTheme) + AlertDialog addDialog = new AlertDialog.Builder(ctx,R.style.dialogTheme) .setMessage("Please name the new player:") .setView(dialogView) .setPositiveButton("Create", new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialogInterface, int i) { - if (playerEdit.getText().length()>0) { + RealmResults matchingPlayers = MiceAndMysticsApplication.getRealmInstance().where(Player.class).equalTo("playerName", playerEdit.getText().toString()).findAll(); + if (playerEdit.getText().length() > 0 && matchingPlayers.size() == 0) { MiceAndMysticsApplication.getRealmInstance().beginTransaction(); Player player = MiceAndMysticsApplication.getRealmInstance().createObject(Player.class); player.setPlayerName(playerEdit.getText().toString()); MiceAndMysticsApplication.getRealmInstance().copyToRealmOrUpdate(player); MiceAndMysticsApplication.getRealmInstance().commitTransaction(); - } else { - Toast.makeText(CampaignDetailsActivity.this,"Can not have a blank name!",Toast.LENGTH_LONG).show(); + } else if (playerEdit.getText().length() == 0) { + Toast.makeText(ctx, "Can not have a blank name!", Toast.LENGTH_LONG).show(); + } else if (matchingPlayers.size() > 0) { + Toast.makeText(ctx, "Can not have same name as another player!", Toast.LENGTH_LONG).show(); } } }) @@ -82,6 +100,12 @@ public class CampaignDetailsActivity extends RecyclerViewActivity { addDialog.show(); } + @Override + protected void onResume() { + super.onResume(); + getAdapter().notifyDataSetChanged(); + } + public void newCharacterDialog() { LayoutInflater inflator = (LayoutInflater)(new ContextThemeWrapper(this, R.style.dialogTheme)).getSystemService(Context.LAYOUT_INFLATER_SERVICE); final View dialogView = inflator.inflate(R.layout.dialog_new_character, null, false); diff --git a/MiceAndMysticsTracker/src/main/java/com/eyecreate/miceandmystics/miceandmystics/PlayerManagerActivity.java b/MiceAndMysticsTracker/src/main/java/com/eyecreate/miceandmystics/miceandmystics/PlayerManagerActivity.java new file mode 100644 index 0000000..0dffc2a --- /dev/null +++ b/MiceAndMysticsTracker/src/main/java/com/eyecreate/miceandmystics/miceandmystics/PlayerManagerActivity.java @@ -0,0 +1,43 @@ +package com.eyecreate.miceandmystics.miceandmystics; + +import android.os.Bundle; +import android.support.v7.widget.LinearLayoutManager; +import android.view.Menu; +import android.view.MenuItem; +import com.eyecreate.miceandmystics.miceandmystics.adapters.PlayerManagerAdapter; + +public class PlayerManagerActivity extends RecyclerViewActivity { + @Override + protected void onCreate(Bundle savedInstanceState) { + super.onCreate(savedInstanceState); + setTitle("Manage Players"); + getSupportActionBar().setDisplayHomeAsUpEnabled(true); + setLayoutManager(new LinearLayoutManager(this)); + setAdapter(new PlayerManagerAdapter()); + } + + @Override + public boolean onCreateOptionsMenu(Menu menu) { + // Inflate the menu; this adds items to the action bar if it is present. + getMenuInflater().inflate(R.menu.menu_player_manager, menu); + return true; + } + + @Override + public boolean onOptionsItemSelected(MenuItem item) { + // Handle action bar item clicks here. The action bar will + // automatically handle clicks on the Home/Up button, so long + // as you specify a parent activity in AndroidManifest.xml. + int id = item.getItemId(); + + //noinspection SimplifiableIfStatement + if (id == R.id.action_add_player) { + CampaignDetailsActivity.newPlayerDialog(this); + return true; + } else if (id == android.R.id.home) { + finish(); + } + + return super.onOptionsItemSelected(item); + } +} diff --git a/MiceAndMysticsTracker/src/main/java/com/eyecreate/miceandmystics/miceandmystics/adapters/PlayerManagerAdapter.java b/MiceAndMysticsTracker/src/main/java/com/eyecreate/miceandmystics/miceandmystics/adapters/PlayerManagerAdapter.java new file mode 100644 index 0000000..4d252b6 --- /dev/null +++ b/MiceAndMysticsTracker/src/main/java/com/eyecreate/miceandmystics/miceandmystics/adapters/PlayerManagerAdapter.java @@ -0,0 +1,92 @@ +package com.eyecreate.miceandmystics.miceandmystics.adapters; + +import android.app.AlertDialog; +import android.content.Context; +import android.content.DialogInterface; +import android.support.v7.widget.RecyclerView; +import android.view.LayoutInflater; +import android.view.ViewGroup; +import com.eyecreate.miceandmystics.miceandmystics.MiceAndMysticsApplication; +import com.eyecreate.miceandmystics.miceandmystics.R; +import com.eyecreate.miceandmystics.miceandmystics.model.*; +import com.eyecreate.miceandmystics.miceandmystics.model.Character; +import com.eyecreate.miceandmystics.miceandmystics.viewholders.PlayerViewHolder; +import io.realm.RealmResults; + +import java.util.Iterator; + +public class PlayerManagerAdapter extends RecyclerView.Adapter { + + LayoutInflater inflater; + RealmResults playerList; + + public PlayerManagerAdapter() { + playerList = MiceAndMysticsApplication.getRealmInstance().where(Player.class).findAll(); + } + + @Override + public PlayerViewHolder onCreateViewHolder(ViewGroup parent, int viewType) { + if(inflater == null) inflater = (LayoutInflater)parent.getContext().getSystemService(Context.LAYOUT_INFLATER_SERVICE); + return new PlayerViewHolder(inflater.inflate(R.layout.item_player,parent,false),this); + } + + @Override + public void onBindViewHolder(PlayerViewHolder holder, int position) { + holder.bindHolder(playerList.get(position)); + } + + @Override + public int getItemCount() { + return playerList.size(); + } + + public void removePlayer(final String playerName,Context ctx) { + //This part checks to see if any characters are attached to this player. + final RealmResults playerCharacters = MiceAndMysticsApplication.getRealmInstance().where(com.eyecreate.miceandmystics.miceandmystics.model.Character.class) + .equalTo("controllingPlayer.playerName", playerName).findAll(); + if(playerCharacters.size()>0) { + AlertDialog removeDialog = new AlertDialog.Builder(ctx,R.style.dialogTheme) + .setMessage("Removing player "+playerName+" must first remove all characters owned by player. Do you want to continue still?") + .setNegativeButton("No", new DialogInterface.OnClickListener() { + @Override + public void onClick(DialogInterface dialogInterface, int i) { + + } + }) + .setPositiveButton("Yes", new DialogInterface.OnClickListener() { + @Override + public void onClick(DialogInterface dialogInterface, int i) { + for(Iterator iter = playerCharacters.iterator();playerCharacters.iterator().hasNext();) { + CampaignDetailsAdapter.removeCharacterFromDB((Character)iter.next()); + } + MiceAndMysticsApplication.getRealmInstance().beginTransaction(); + MiceAndMysticsApplication.getRealmInstance().where(Player.class).equalTo("playerName",playerName).findFirst().removeFromRealm(); + MiceAndMysticsApplication.getRealmInstance().commitTransaction(); + notifyDataSetChanged(); + } + }) + .create(); + removeDialog.show(); + } else { + AlertDialog removeDialog = new AlertDialog.Builder(ctx, R.style.dialogTheme) + .setMessage("Do you want to remove player: " + playerName + "?") + .setNegativeButton("No", new DialogInterface.OnClickListener() { + @Override + public void onClick(DialogInterface dialogInterface, int i) { + + } + }) + .setPositiveButton("Yes", new DialogInterface.OnClickListener() { + @Override + public void onClick(DialogInterface dialogInterface, int i) { + MiceAndMysticsApplication.getRealmInstance().beginTransaction(); + MiceAndMysticsApplication.getRealmInstance().where(Player.class).equalTo("playerName",playerName).findFirst().removeFromRealm(); + MiceAndMysticsApplication.getRealmInstance().commitTransaction(); + notifyDataSetChanged(); + } + }) + .create(); + removeDialog.show(); + } + } +} diff --git a/MiceAndMysticsTracker/src/main/java/com/eyecreate/miceandmystics/miceandmystics/viewholders/PlayerViewHolder.java b/MiceAndMysticsTracker/src/main/java/com/eyecreate/miceandmystics/miceandmystics/viewholders/PlayerViewHolder.java new file mode 100644 index 0000000..971286f --- /dev/null +++ b/MiceAndMysticsTracker/src/main/java/com/eyecreate/miceandmystics/miceandmystics/viewholders/PlayerViewHolder.java @@ -0,0 +1,31 @@ +package com.eyecreate.miceandmystics.miceandmystics.viewholders; + +import android.support.v7.widget.RecyclerView; +import android.view.View; +import android.widget.TextView; +import com.eyecreate.miceandmystics.miceandmystics.R; +import com.eyecreate.miceandmystics.miceandmystics.adapters.PlayerManagerAdapter; +import com.eyecreate.miceandmystics.miceandmystics.model.Player; + +public class PlayerViewHolder extends RecyclerView.ViewHolder implements View.OnLongClickListener { + + TextView playerName; + PlayerManagerAdapter adapter; + + public PlayerViewHolder(View itemView,PlayerManagerAdapter adapter) { + super(itemView); + this.adapter = adapter; + playerName = (TextView)itemView.findViewById(R.id.player_name); + itemView.setOnLongClickListener(this); + } + + public void bindHolder(Player player) { + playerName.setText(player.getPlayerName()); + } + + @Override + public boolean onLongClick(View view) { + adapter.removePlayer(playerName.getText().toString(),view.getContext()); + return true; + } +} diff --git a/MiceAndMysticsTracker/src/main/res/layout/item_player.xml b/MiceAndMysticsTracker/src/main/res/layout/item_player.xml new file mode 100644 index 0000000..69bca3e --- /dev/null +++ b/MiceAndMysticsTracker/src/main/res/layout/item_player.xml @@ -0,0 +1,27 @@ + + + + + + + + \ No newline at end of file diff --git a/MiceAndMysticsTracker/src/main/res/menu/menu_campaign_details.xml b/MiceAndMysticsTracker/src/main/res/menu/menu_campaign_details.xml index 05c4ad0..f5e2317 100644 --- a/MiceAndMysticsTracker/src/main/res/menu/menu_campaign_details.xml +++ b/MiceAndMysticsTracker/src/main/res/menu/menu_campaign_details.xml @@ -9,5 +9,13 @@ + + diff --git a/MiceAndMysticsTracker/src/main/res/menu/menu_player_manager.xml b/MiceAndMysticsTracker/src/main/res/menu/menu_player_manager.xml new file mode 100644 index 0000000..72b54b8 --- /dev/null +++ b/MiceAndMysticsTracker/src/main/res/menu/menu_player_manager.xml @@ -0,0 +1,8 @@ + + + + \ No newline at end of file diff --git a/MiceAndMysticsTracker/src/main/res/mipmap-xxhdpi/ic_menu_friendslist.png b/MiceAndMysticsTracker/src/main/res/mipmap-xxhdpi/ic_menu_friendslist.png new file mode 100644 index 0000000000000000000000000000000000000000..920d687e5cd985e91d2da3e1622f48230a9670c4 GIT binary patch literal 3228 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 S_Fm-X|AMu-1GWn7AO9~{$yosa literal 0 HcmV?d00001 diff --git a/MiceAndMysticsTracker/src/main/res/mipmap-xxhdpi/ic_menu_star.png b/MiceAndMysticsTracker/src/main/res/mipmap-xxhdpi/ic_menu_star.png new file mode 100644 index 0000000000000000000000000000000000000000..0b9af8e104d00b65271602ec21c0bd88c4474428 GIT binary patch literal 3534 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 literal 0 HcmV?d00001 -- GitLab