package org.bukkit.craftbukkit.inventory; import java.util.ArrayList; import java.util.List; import java.util.Map; import net.minecraft.server.NBTTagCompound; import net.minecraft.server.NBTTagList; import org.apache.commons.lang.Validate; import org.bukkit.Material; import org.bukkit.configuration.serialization.DelegateDeserialization; import org.bukkit.craftbukkit.inventory.CraftMetaItem.SerializableMeta; import org.bukkit.inventory.meta.BookMeta; import com.google.common.base.Strings; import com.google.common.collect.ImmutableList; import com.google.common.collect.ImmutableMap.Builder; import java.util.AbstractList; import net.minecraft.server.ChatSerializer; import net.minecraft.server.IChatBaseComponent; import net.minecraft.server.NBTTagString; import org.bukkit.craftbukkit.util.CraftChatMessage; @DelegateDeserialization(SerializableMeta.class) class CraftMetaBook extends CraftMetaItem implements BookMeta { static final ItemMetaKey BOOK_TITLE = new ItemMetaKey("title"); static final ItemMetaKey BOOK_AUTHOR = new ItemMetaKey("author"); static final ItemMetaKey BOOK_PAGES = new ItemMetaKey("pages"); static final ItemMetaKey RESOLVED = new ItemMetaKey("resolved"); static final ItemMetaKey GENERATION = new ItemMetaKey("generation"); static final int MAX_PAGE_LENGTH = Short.MAX_VALUE; // TODO: Check me static final int MAX_TITLE_LENGTH = 0xffff; protected String title; protected String author; protected List pages = new ArrayList(); protected Integer generation; CraftMetaBook(CraftMetaItem meta) { super(meta); if (meta instanceof CraftMetaBook) { CraftMetaBook bookMeta = (CraftMetaBook) meta; this.title = bookMeta.title; this.author = bookMeta.author; pages.addAll(bookMeta.pages); this.generation = bookMeta.generation; } } CraftMetaBook(NBTTagCompound tag) { this(tag, true); } CraftMetaBook(NBTTagCompound tag, boolean handlePages) { super(tag); if (tag.hasKey(BOOK_TITLE.NBT)) { this.title = tag.getString(BOOK_TITLE.NBT); } if (tag.hasKey(BOOK_AUTHOR.NBT)) { this.author = tag.getString(BOOK_AUTHOR.NBT); } boolean resolved = false; if (tag.hasKey(RESOLVED.NBT)) { resolved = tag.getBoolean(RESOLVED.NBT); } if (tag.hasKey(GENERATION.NBT)) { generation = tag.getInt(GENERATION.NBT); } if (tag.hasKey(BOOK_PAGES.NBT) && handlePages) { NBTTagList pages = tag.getList(BOOK_PAGES.NBT, 8); for (int i = 0; i < pages.size(); i++) { String page = pages.getString(i); if (resolved) { try { this.pages.add(ChatSerializer.a(page)); continue; } catch (Exception e) { // Ignore and treat as an old book } } addPage(page); } } } CraftMetaBook(Map map) { super(map); setAuthor(SerializableMeta.getString(map, BOOK_AUTHOR.BUKKIT, true)); setTitle(SerializableMeta.getString(map, BOOK_TITLE.BUKKIT, true)); Iterable pages = SerializableMeta.getObject(Iterable.class, map, BOOK_PAGES.BUKKIT, true); if(pages != null) { for (Object page : pages) { if (page instanceof String) { addPage((String) page); } } } generation = SerializableMeta.getObject(Integer.class, map, GENERATION.BUKKIT, true); } @Override void applyToItem(NBTTagCompound itemData) { applyToItem(itemData, true); } void applyToItem(NBTTagCompound itemData, boolean handlePages) { super.applyToItem(itemData); if (hasTitle()) { itemData.setString(BOOK_TITLE.NBT, this.title); } if (hasAuthor()) { itemData.setString(BOOK_AUTHOR.NBT, this.author); } if (handlePages) { if (hasPages()) { NBTTagList list = new NBTTagList(); for (IChatBaseComponent page : pages) { list.add(new NBTTagString(CraftChatMessage.fromComponent(page))); } itemData.set(BOOK_PAGES.NBT, list); } itemData.remove(RESOLVED.NBT); } if (generation != null) { itemData.setInt(GENERATION.NBT, generation); } } @Override boolean isEmpty() { return super.isEmpty() && isBookEmpty(); } boolean isBookEmpty() { return !(hasPages() || hasAuthor() || hasTitle()); } @Override boolean applicableTo(Material type) { switch (type) { case WRITTEN_BOOK: case BOOK_AND_QUILL: return true; default: return false; } } public boolean hasAuthor() { return !Strings.isNullOrEmpty(author); } public boolean hasTitle() { return !Strings.isNullOrEmpty(title); } public boolean hasPages() { return !pages.isEmpty(); } public String getTitle() { return this.title; } public boolean setTitle(final String title) { if (title == null) { this.title = null; return true; } else if (title.length() > MAX_TITLE_LENGTH) { return false; } this.title = title; return true; } public String getAuthor() { return this.author; } public void setAuthor(final String author) { this.author = author; } public String getPage(final int page) { Validate.isTrue(isValidPage(page), "Invalid page number"); return CraftChatMessage.fromComponent(pages.get(page - 1)); } public void setPage(final int page, final String text) { if (!isValidPage(page)) { throw new IllegalArgumentException("Invalid page number " + page + "/" + pages.size()); } String newText = text == null ? "" : text.length() > MAX_PAGE_LENGTH ? text.substring(0, MAX_PAGE_LENGTH) : text; pages.set(page - 1, CraftChatMessage.fromString(newText, true)[0]); } public void setPages(final String... pages) { this.pages.clear(); addPage(pages); } public void addPage(final String... pages) { for (String page : pages) { if (page == null) { page = ""; } else if (page.length() > MAX_PAGE_LENGTH) { page = page.substring(0, MAX_PAGE_LENGTH); } this.pages.add(CraftChatMessage.fromString(page, true)[0]); } } public int getPageCount() { return pages.size(); } public List getPages() { final List copy = ImmutableList.copyOf(pages); return new AbstractList() { @Override public String get(int index) { return CraftChatMessage.fromComponent(copy.get(index)); } @Override public int size() { return copy.size(); } }; } public void setPages(List pages) { this.pages.clear(); for (String page : pages) { addPage(page); } } private boolean isValidPage(int page) { return page > 0 && page <= pages.size(); } @Override public CraftMetaBook clone() { CraftMetaBook meta = (CraftMetaBook) super.clone(); meta.pages = new ArrayList(pages); return meta; } @Override int applyHash() { final int original; int hash = original = super.applyHash(); if (hasTitle()) { hash = 61 * hash + this.title.hashCode(); } if (hasAuthor()) { hash = 61 * hash + 13 * this.author.hashCode(); } if (hasPages()) { hash = 61 * hash + 17 * this.pages.hashCode(); } return original != hash ? CraftMetaBook.class.hashCode() ^ hash : hash; } @Override boolean equalsCommon(CraftMetaItem meta) { if (!super.equalsCommon(meta)) { return false; } if (meta instanceof CraftMetaBook) { CraftMetaBook that = (CraftMetaBook) meta; return (hasTitle() ? that.hasTitle() && this.title.equals(that.title) : !that.hasTitle()) && (hasAuthor() ? that.hasAuthor() && this.author.equals(that.author) : !that.hasAuthor()) && (hasPages() ? that.hasPages() && this.pages.equals(that.pages) : !that.hasPages()); } return true; } @Override boolean notUncommon(CraftMetaItem meta) { return super.notUncommon(meta) && (meta instanceof CraftMetaBook || isBookEmpty()); } @Override Builder serialize(Builder builder) { super.serialize(builder); if (hasTitle()) { builder.put(BOOK_TITLE.BUKKIT, title); } if (hasAuthor()) { builder.put(BOOK_AUTHOR.BUKKIT, author); } if (hasPages()) { List pagesString = new ArrayList(); for (IChatBaseComponent comp : pages) { pagesString.add(CraftChatMessage.fromComponent(comp)); } builder.put(BOOK_PAGES.BUKKIT, pagesString); } if (generation != null) { builder.put(GENERATION.BUKKIT, generation); } return builder; } }