24
18

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 5 years have passed since last update.

マイクラプログラミングの世界Advent Calendar 2015

Day 11

CraftBukkit・Spigotプラグインを作るときの小技のような何か

Last updated at Posted at 2015-12-03

MinecraftのサーバMOD CraftBukkit・Spigotのプラグイン
1.7以前に作ったものなのでやり方が古いかも。
小技と言えないのもあるし数が少ないけど、何かしら役に立てば良いかなと思う(:3_ヽ)_
間違ってたり、もっと良い方法があったり、分からない部分があったらコメントに書くかTwitterにリプお願いします。

#・Tab押下時に表示されるリストの名前を変更
1番目から小技っぽくないけど。
Player#setPlayerListName(nameString)
チームや権限で色を分けるなどに使える。
名前を全く違う文字列にもできるけど、文字数の上限は16文字なので注意が必要。
17文字以上にすると例外が吐かれる。

※色を付ける場合、色の部分で2文字使われている。
ChatColor.RED + "Name"
  ↓
"§cName"

色コード Formatting codes

うちの1.8.8テスト環境だと何故か上限突破できる。
Javadocには16文字上限って書いてあるけど、どこかのバージョンで変更されたのかな・・・。
ListNameChange.jpg

#・登録したイベントを最後に呼んでもらう
他のプラグインで、ブロックやアイテムの変更があるかもしれない時に、そのプラグインがイベントを処理してから、自分のプラグインの処理を実行したい場合など。

イベントリスナをPluginManager#registerEvents()するタイミングを遅らせるだけ。

設定したEventPriorityの中で最後になるだけなので、EventPriority.NORMAL設定でEventPriority.MONITORより後に呼ばれるわけではない。

色々方法はあるけど、とりあえずスケジューラのだけ載せておく。
初期化コマンドを作るとかで、任意のタイミングで実行できる方が確実かも。
BukkitScheduler#scheduleSyncDelayedTask(plugin, task, delay)

useScheduler.java
Bukkit.getServer().getScheduler().scheduleSyncDelayedTask(plugin, () -> {
	Bukkit.getServer().getPluginManager().registerEvents(plugin, listener);
}, 200);

3番目の引数はTick。

#・コマンド入力時のTabキー補完
これも小技ではない気がする。
/test hoge, /test piyoなどのコマンドが用意されていたら
/test時点でTabキーを押すとチャット欄にhoge, piyoが表示され1番目が入力欄に入り、更にTabキーで表示されている候補を回すことができ、
/test hでTabキーを押すと入力欄に残りのogeが補完されるやつ。

TabCompleterを実装したクラスで、onTabComplete()をオーバーライドして使う。
JavaPluginには実装されているのでメインクラスにはすぐ書ける。
JavaPlugin#getCommand(commandString).setExecutor(commandExecutor)でコマンド別のクラスで処理する場合、CommandExecutorを実装しているそのクラスにTabCompleterを実装する。

上で書いた2つのコマンド補完

tabComplete.java
public class Main extends JavaPlugin {
	@Override
	public List<String> onTabComplete(CommandSender sender, Command command, String alias, String[] args) {
		if (!command.getName().equalsIgnoreCase("test")) return super.onTabComplete(sender, command, alias, args);
		if (args.length == 1) {
			if (args[0].length() == 0) { // /testまで
				return Arrays.asList("hoge", "piyo");
			} else {
				//入力されている文字列と先頭一致
				if ("hoge".startsWith(args[0])) {
					return Collections.singletonList("hoge");
				} else if ("piyo".startsWith(args[0])) {
					return Collections.singletonList("piyo");
				}
			}
		}
		//JavaPlugin#onTabComplete()を呼び出す
		return super.onTabComplete(sender, command, alias, args);
	}
}

#・看板の文字を更新
看板はとても有用。
プレイヤーのログイン情報を出したり、看板をクリックで何か動作をしたり。
Sign
SignChangeEvent
下側のイベントは例でPlayerInteractEventにしたけど、看板ブロックを取得できればWorldから座標指定して直接取得しても良いと思う。

signChange.java
@EventHandler
public void onSignChange(SignChangeEvent event) {
	String[] lines = event.getLines();//全行取得
	String line = event.getLine(0);//引数は[0-3]

	event.setLine(0, "0行目です");//行に文字列をセット

	//SignChangeEventではイベントに値をセットするのでupdate不要
}

@EventHandler //看板ブロックを右クリック
public void onSignClick(PlayerInteractEvent event) {
	if (event.getAction() != Action.RIGHT_CLICK_BLOCK) return;
	Block clickedBlock = event.getClickedBlock();
	Material material = clickedBlock.getType();
	if (material == Material.SIGN_POST || material == Material.WALL_SIGN) {
		Sign sign = (Sign) clickedBlock.getState();

		String[] lines = sign.getLines();//全行取得
		String line = sign.getLine(0);//引数は[0-3]

		sign.setLine(0, "0行目です");//行に文字列をセット

		sign.update(); //Signを直接変更する場合はこれで更新完了
		//updateを忘れると反映されないので注意
	}
}

##応用?
PlayerItemHeldEventでプレイヤーがスクロール操作した時のスロット(何て名称か分からない)の選択変更を検知できる。(数字キーでも呼ばれる)
このイベントから変更前・変更後のスロット番号を取得できるので、プレイヤーが上下どちらにスクロールしたかを判定して、上下への選択項目変更などに使える。

スクロール検知は看板以外でも使えるけど、看板変更の応用なのでそのコードを載せておく(:3_ヽ)_

・プレイヤーが看板に向かって、カーソル(画面中央の十字)を合わせているときにスクロールすると、看板の文字が変化する。
・設置してある看板には既に文字列が入ってる状態
ChatColor.GREEN + "Item 0"
"Item 1"
"Item 2"
"Item 3"

scrollSign.java
private int currentLine = 0;

@EventHandler
public void onScrolling(PlayerItemHeldEvent event){
	//プレイヤーの向いている方向にあるブロックを取得
	Block block = event.getPlayer().getTargetBlock((Set<Material>) null, 5);
	Material material = block.getType();
	if (material == Material.SIGN_POST || material == Material.WALL_SIGN) {
		Sign sign = (Sign) block.getState();
		//変更後、変更前のスロット番号取得
		//手持ちスロットの番号は[0-8]
		int slotIndex = event.getNewSlot() - event.getPreviousSlot();
		if(slotIndex == -1 || slotIndex == 8){
			//up
			selectLine(sign, true);
		}else if(slotIndex == 1 || slotIndex == -8){
			//down
			selectLine(sign, false);
		}
	}
}

private void selectLine(Sign sign, boolean scrollUp) {
	if (scrollUp && currentLine == 0) {//0行目状態で上スクロール取消
		return;
	}
	if (!scrollUp && currentLine == 3) {//3行目状態で下スクロール取消
		return;
	}
	String line = sign.getLine(currentLine);
	line = line.substring(2);//先頭の色コード削除
	sign.setLine(currentLine, line);
	if (scrollUp) {
		currentLine--;
	} else {
		currentLine++;
	}
	line = sign.getLine(currentLine);
	line = ChatColor.GREEN + line;//先頭に色コード追加
	sign.setLine(currentLine, line);
	sign.update();
}

ある程度見やすいように書いたつもりだけど、分かりにくかったらごめんなさい。
scrollSign.jpg
画像は「Item 0」の行が選択された状態で、スロット[1]からスロット[2]に変更したもの。
横の穴は気にしないで。

あとはこれを使って色々機能を作る感じ。
「看板右クリックしたら、選択されてる行に書いてるアイテムを出す」とか。

#・簡易カスタムアイテム
アイテムの名前を変更したり、説明文を追加したり。
ItemStack#getItemMeta() で取得できるItemMetaからアイテム名などを変更できる。

itemMetaChange.java
ItemStack itemStack = new ItemStack(Material.BREAD);
ItemMeta itemMeta = itemStack.getItemMeta();
//アイテム名をセット
itemMeta.setDisplayName("焼きたてのパン");
List<String> lore = Arrays.asList(ChatColor.RESET + "とても良い香りがする", ChatColor.RESET + "120円");
//説明文をセット
itemMeta.setLore(lore);
//ItemMetaをセットしないと反映されない。
itemStack.setItemMeta(itemMeta);

説明文の所は文字列の中で改行ではなく、Listに要素を追加することで改行することができる。
文字列の先頭にChatColor.RESETを入れ無い場合、画像の下側のように先頭に斜体と紫色のコードが入る。
DisplayNameは先頭に斜体のみ
Loreは先頭に斜体と紫色のコード(ChatColor.DARK_PURPLE)
2015-12-03_23.32.48.png

#・インベントリ作成
鞄・バックパックなど追加収納を作りたい時に。
Server#createInventory()
Inventory inventory = Bukkit.getServer().createInventory(null, 9 * row, "バックパック")
2番目の引数は必ず9の倍数。
このメソッドで作るとInventoryTypeはCHESTで作られる。
新しく作ったインベントリをプレイヤーに開かせるには、HumanEntity#openInventory()に作ったインベントリを渡す。
渡すとすぐ開かれるので注意。

画像は2番目の引数を27で実行したので3行だが、9を渡せば1行にもできる。
createInventory.jpg
※注意
作ったインベントリはサーバに保存ません。
インベントリ名、スロット数、中身などの保存は自分で実装する必要があります。
BukkitObjectOutputStreamと本体にあるBase64Coderでシリアライズするのが一般的のはず・・・。

##応用
収納以外で使ってみる。
アイテム名を変えたアイテムを置き、それをクリックすることで何かを動作させるUIに。
新しく作ったインベントリでもInventoryClickEventが呼ばれるので、そこでイベントをキャンセルしてクリックされたアイテムを判定することで実装できる。(要インベントリ整理MOD対処)
InventoryClickEvent#getSlotType()がSlotType.OUTSIDE(グレーアウトしてる部分)だったら次のページへ移動なども。
イベントが呼ばれた時に、アイテムをカーソルが掴んでいる常態か、スロットにある状態か迷いやすいので注意。
InventoryClickEvent#getCursor()
InventoryClickEvent#getCurrentItem()

#(:3_ヽ)_
昔のソースを漁って「使えそうなの6個だけか・・・10個くらいあればな」と思ってたけど、意外と長くなってしまった。
役に立ったのがあればいいな。

24
18
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
24
18

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?