diff --git a/xyz.veronie.bgg.ui/Application.e4xmi b/xyz.veronie.bgg.ui/Application.e4xmi index 32eaa62..002cfe8 100644 --- a/xyz.veronie.bgg.ui/Application.e4xmi +++ b/xyz.veronie.bgg.ui/Application.e4xmi @@ -7,7 +7,13 @@ View - + + + + + + + @@ -15,10 +21,15 @@ - - + + + + + + + diff --git a/xyz.veronie.bgg.ui/icons/noun_Save_16x16.png b/xyz.veronie.bgg.ui/icons/noun_Save_16x16.png new file mode 100644 index 0000000..440a12e Binary files /dev/null and b/xyz.veronie.bgg.ui/icons/noun_Save_16x16.png differ diff --git a/xyz.veronie.bgg.ui/icons/noun_import_16x16.png b/xyz.veronie.bgg.ui/icons/noun_import_16x16.png new file mode 100644 index 0000000..72c5494 Binary files /dev/null and b/xyz.veronie.bgg.ui/icons/noun_import_16x16.png differ diff --git a/xyz.veronie.bgg.ui/icons/noun_import_60x60.png b/xyz.veronie.bgg.ui/icons/noun_import_60x60.png new file mode 100644 index 0000000..b1836fc Binary files /dev/null and b/xyz.veronie.bgg.ui/icons/noun_import_60x60.png differ diff --git a/xyz.veronie.bgg.ui/icons/noun_open_16x16.png b/xyz.veronie.bgg.ui/icons/noun_open_16x16.png new file mode 100644 index 0000000..8b276f1 Binary files /dev/null and b/xyz.veronie.bgg.ui/icons/noun_open_16x16.png differ diff --git a/xyz.veronie.bgg.ui/plugin.xml b/xyz.veronie.bgg.ui/plugin.xml index 4a77977..9cf6b9b 100644 --- a/xyz.veronie.bgg.ui/plugin.xml +++ b/xyz.veronie.bgg.ui/plugin.xml @@ -2,18 +2,6 @@ - - - - - - diff --git a/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/localdb/LocalDbAdapterService.java b/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/localdb/LocalDbAdapterService.java index 76cbbc0..3c0e042 100644 --- a/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/localdb/LocalDbAdapterService.java +++ b/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/localdb/LocalDbAdapterService.java @@ -36,6 +36,5 @@ public class LocalDbAdapterService { public List loadThingListNames() throws SQLException { return sqliteController.getThingLists(); } - } diff --git a/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/localdb/SqliteController.java b/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/localdb/SqliteController.java index 2ab7f5b..fa2e1a7 100644 --- a/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/localdb/SqliteController.java +++ b/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/localdb/SqliteController.java @@ -173,7 +173,7 @@ public class SqliteController { PreparedStatement thingStatement = connection .prepareStatement("INSERT OR REPLACE INTO Thing VALUES (?, ?, ?, ?, ?, ?);"); PreparedStatement listToThingStatement = connection - .prepareStatement("INSERT INTO ThingListToThing VALUES (?, ?);"); + .prepareStatement("INSERT OR IGNORE INTO ThingListToThing VALUES (?, ?);"); for (Thing thing : things) { ThingMetaData metaData = thing.getMetaData(); @@ -266,4 +266,5 @@ public class SqliteController { } + } diff --git a/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/result/BggApi.java b/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/result/BggApi.java index 7600914..7b0ceef 100644 --- a/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/result/BggApi.java +++ b/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/result/BggApi.java @@ -12,6 +12,7 @@ import java.net.URLEncoder; import java.util.ArrayList; import java.util.Iterator; import java.util.Map.Entry; +import java.util.Set; import javax.inject.Inject; import javax.xml.parsers.DocumentBuilder; @@ -61,6 +62,21 @@ public class BggApi { } } + /// retrieve a list of things from BGG with the given ids + public ArrayList getThings(Set ids) { + StringBuilder urlStr = new StringBuilder(); + urlStr.append(BASE_URL + "thing?id="); + + String sep = ""; + for (Integer integer : ids) { + urlStr.append(sep + integer.toString()); + if(sep.isEmpty()) { + sep = ","; + } + } + + return getThings(urlStr.toString()); + } public ArrayList getThingsForUser(String user) throws IllegalArgumentException { @@ -204,7 +220,7 @@ public class BggApi { if (nNode.getNodeType() == Node.ELEMENT_NODE) { Element eElement = (Element) nNode; - // thing + // thing when fetching by user if( eElement.hasAttribute("objecttype") && eElement.getAttribute("objecttype").equals("thing")) { @@ -216,12 +232,45 @@ public class BggApi { getValue(eElement, "image"), getValue(eElement, "thumbnail"), getValue(eElement, "comment"), - Integer.parseInt(getValue(eElement, "numplays")) + Integer.parseInt(getValue(eElement, "numplays")) ); - Thing thing = new Thing(id, tmd); + Thing thing = new Thing(id, tmd); things.add(thing); } } + // when fetching things by id, type is boardgame + else if(eElement.hasAttribute("type") && + eElement.getAttribute("type").equals("boardgame")) + { + Integer id = Integer.parseInt(eElement.getAttribute("id")); + if(id != 0) { + String name = ""; + NodeList nameList = eElement.getElementsByTagName("name"); + for(int n = 0; n < nameList.getLength(); n++) { + Node nameNode = nameList.item(n); + if (nNode.getNodeType() == Node.ELEMENT_NODE) { + Element nameElement = (Element) nameNode; + if(nameElement.hasAttribute("type") + && nameElement.getAttribute("type").equals("primary")) + { + name = nameElement.getAttribute("value"); + break; + } + } + } + + ThingMetaData tmd = new ThingMetaData( + id, + name, + getValue(eElement, "image"), + getValue(eElement, "thumbnail"), + getValue(eElement, "comment"), + null + ); + Thing thing = new Thing(id, tmd); + things.add(thing); + } + } // family has "type" else if(eElement.hasAttribute("type") && eElement.getAttribute("type").equals("boardgamefamily")) @@ -312,6 +361,7 @@ public class BggApi { return null; } + private void checkForErrors(Document doc) throws IllegalArgumentException { NodeList nList = doc.getElementsByTagName("error"); if(nList.getLength() > 0) { diff --git a/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/result/ThingProvider.java b/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/result/ThingProvider.java index 5966b13..c5ae7eb 100644 --- a/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/result/ThingProvider.java +++ b/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/result/ThingProvider.java @@ -97,6 +97,13 @@ public class ThingProvider { // TODO: handle unsaved current list this.things = localDbAdapterService.loadThingList(name); } + + /// store things as a new list + public void importList(List importThings, String listName) { + if(importThings != null && !importThings.isEmpty()) { + this.things = importThings; + } + } public int numberOfThings() { return things.size(); diff --git a/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/ui/dialogs/LoadGameListDialog.java b/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/ui/dialogs/LoadGameListDialog.java index d93b044..74a037b 100644 --- a/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/ui/dialogs/LoadGameListDialog.java +++ b/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/ui/dialogs/LoadGameListDialog.java @@ -35,8 +35,11 @@ public class LoadGameListDialog extends Dialog { /** * Create the dialog. * @param parentShell + * @param handlerService + * @param commandService */ - public LoadGameListDialog(Shell parentShell, List thingLists) { + public LoadGameListDialog(Shell parentShell, List thingLists) + { super(parentShell); this.thingLists = thingLists; setShellStyle(SWT.BORDER | SWT.RESIZE | SWT.APPLICATION_MODAL); @@ -44,6 +47,7 @@ public class LoadGameListDialog extends Dialog { /** * Create contents of the dialog. + * * @param parent */ @Override @@ -112,7 +116,11 @@ public class LoadGameListDialog extends Dialog { }); column.pack(); - + + Composite otherActionsComposite = new Composite(container, SWT.NONE); + otherActionsComposite.setLayoutData(new GridData(SWT.RIGHT, SWT.CENTER, false, false, 1, 1)); + otherActionsComposite.setLayout(new GridLayout(1, false)); + return container; } @@ -121,8 +129,8 @@ public class LoadGameListDialog extends Dialog { this.setReturnCode(OK); this.close(); } - + /** * Create contents of the button bar. * @param parent @@ -146,5 +154,5 @@ public class LoadGameListDialog extends Dialog { public String getSelectedName() { return selectedName; } - + } diff --git a/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/ui/handlers/ImportResultTxtHandler.java b/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/ui/handlers/ImportResultTxtHandler.java new file mode 100644 index 0000000..f368864 --- /dev/null +++ b/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/ui/handlers/ImportResultTxtHandler.java @@ -0,0 +1,134 @@ + +package xyz.veronie.bgg.ui.handlers; + +import java.io.BufferedReader; +import java.io.FileNotFoundException; +import java.io.FileReader; +import java.io.IOException; +import java.util.HashSet; +import java.util.List; +import java.util.Set; + +import org.eclipse.core.runtime.Path; +import org.eclipse.e4.core.di.annotations.CanExecute; +import org.eclipse.e4.core.di.annotations.Execute; +import org.eclipse.e4.core.services.events.IEventBroker; +import org.eclipse.swt.SWT; +import org.eclipse.swt.widgets.FileDialog; +import org.eclipse.swt.widgets.MessageBox; +import org.eclipse.swt.widgets.Shell; + +import xyz.veronie.bgg.result.BggApi; +import xyz.veronie.bgg.result.Thing; +import xyz.veronie.bgg.result.ThingProvider; +import xyz.veronie.bgg.types.EventConstants; + +public class ImportResultTxtHandler { + + private Shell shell; + + + @Execute + public void execute(Shell shell, ThingProvider thingProvider, BggApi bggApi, IEventBroker eventBroker) { + this.shell = shell; + + FileDialog dialog = new FileDialog(shell, SWT.OPEN); + dialog.setText("Select a result.txt from bgg1tool"); + String[] exfilters = { "*.txt" }; + dialog.setFilterExtensions(exfilters); + String resultPath = dialog.open(); + + if(resultPath != null && !resultPath.isEmpty()) { + Set thingIds = parseResultTxtIds(resultPath); + if(thingIds != null && !thingIds.isEmpty()) { + String listName = new Path(resultPath).lastSegment() + " (unsaved)"; + eventBroker.send(EventConstants.TOPIC_STATUS, "Fetching entries from BGG..."); + List things = bggApi.getThings(thingIds); + thingProvider.importList(things, listName); + eventBroker.send(EventConstants.TOPIC_RESULT_CHANGED, listName); + eventBroker.send(EventConstants.TOPIC_STATUS, "Fetched " + Integer.toString(things.size()) + " things."); + } + } + } + + + private Set parseResultTxtIds(String resultPath) { + FileReader input = null; + String myLine = null; + int lineNo = 0; // count overall lines + int errNo = 0; // count unparsable lines + int dupNo = 0; // count duplicates + try { + input = new FileReader(resultPath); + + BufferedReader bufferedReader = new BufferedReader(input); + + Set ids = new HashSet(); + while ( (myLine = bufferedReader.readLine()) != null) + { + lineNo++; + String[] tokens = myLine.split(","); + if(tokens.length > 0) { + if(tokens[0].equals("id")) continue; // header line + try { + boolean exists = !ids.add(Integer.parseInt(tokens[0])); + if(exists) { + System.out.println("DEBUG: duplicate id: " + tokens[0]); + dupNo++; + } + } + catch (NumberFormatException e) { + errNo++; + System.out.println("WARN: Could not parse id from line " + Integer.toString(lineNo) + + ", line starts with '" + tokens[0] + "'"); + } + } + } + + bufferedReader.close(); + + if(ids != null) { + MessageBox msgBox = new MessageBox(shell, SWT.ICON_INFORMATION | SWT.OK); + StringBuilder msg = new StringBuilder(); + int thingsNo = lineNo - errNo - dupNo - 1; + msg.append("Parsing successful. Found ").append(Integer.toString(thingsNo)).append(" things."); + if(errNo > 0) { + msg.append("\n\rSkipped ").append(Integer.toString(errNo)).append(" line(s)."); + } + if(dupNo > 0) { + msg.append("\n\r").append(Integer.toString(dupNo)).append(" duplicate ids"); + } + msg.append("\n\r"); + msgBox.setMessage(msg.toString()); + msgBox.open(); + } + + System.out.println("TRACE: " + ids); + return ids; + + } catch (FileNotFoundException e) { + MessageBox msgBox = new MessageBox(shell, SWT.ICON_ERROR | SWT.OK); + msgBox.setMessage("Could not open file '" + resultPath + "'."); + msgBox.open(); + e.printStackTrace(); + } catch (IOException e) { + MessageBox msgBox = new MessageBox(shell, SWT.ICON_ERROR | SWT.OK); + String msg = "Error while parsing '" + resultPath + "'.\r\n" + + "It must have comma separated lines starting with an integer number.\r\n"; + msgBox.setMessage(msg); + msgBox.open(); + e.printStackTrace(); + } + + + return null; + } + + + @CanExecute + public boolean canExecute() { + + return true; + } + +} \ No newline at end of file diff --git a/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/ui/handlers/HandleLoadGamelist.java b/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/ui/handlers/LoadGamelistHandler.java similarity index 82% rename from xyz.veronie.bgg.ui/src/xyz/veronie/bgg/ui/handlers/HandleLoadGamelist.java rename to xyz.veronie.bgg.ui/src/xyz/veronie/bgg/ui/handlers/LoadGamelistHandler.java index 112e212..06249f9 100644 --- a/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/ui/handlers/HandleLoadGamelist.java +++ b/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/ui/handlers/LoadGamelistHandler.java @@ -6,6 +6,8 @@ import java.util.List; import javax.inject.Inject; +import org.eclipse.e4.core.commands.ECommandService; +import org.eclipse.e4.core.commands.EHandlerService; import org.eclipse.e4.core.di.annotations.CanExecute; import org.eclipse.e4.core.di.annotations.Execute; import org.eclipse.e4.core.services.events.IEventBroker; @@ -18,13 +20,14 @@ import xyz.veronie.bgg.result.ThingProvider; import xyz.veronie.bgg.types.EventConstants; import xyz.veronie.bgg.ui.dialogs.LoadGameListDialog; -public class HandleLoadGamelist { +@SuppressWarnings("restriction") +public class LoadGamelistHandler { @Inject ThingProvider thingProvider; @Execute - public void execute(Shell shell, IEventBroker eventBroker) { + public void execute(Shell shell, IEventBroker eventBroker, ECommandService commandService, EHandlerService handlerService) { try { List thingLists = thingProvider.getThingListNames(); diff --git a/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/ui/handlers/HandleSaveGamelist.java b/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/ui/handlers/SaveGamelistHandler.java similarity index 89% rename from xyz.veronie.bgg.ui/src/xyz/veronie/bgg/ui/handlers/HandleSaveGamelist.java rename to xyz.veronie.bgg.ui/src/xyz/veronie/bgg/ui/handlers/SaveGamelistHandler.java index 0e4b1aa..bb9abba 100644 --- a/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/ui/handlers/HandleSaveGamelist.java +++ b/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/ui/handlers/SaveGamelistHandler.java @@ -15,19 +15,18 @@ import xyz.veronie.bgg.result.ThingProvider; import xyz.veronie.bgg.types.EventConstants; import xyz.veronie.bgg.ui.dialogs.SaveGameListDialog; -public class HandleSaveGamelist { +public class SaveGamelistHandler { @Inject ThingProvider thingProvider; - public HandleSaveGamelist() { + public SaveGamelistHandler() { } @Execute public void execute(Shell shell, IEventBroker eventBroker) { - // TODO: disable save if list is empty if(thingProvider.numberOfThings() == 0) { MessageBox msgBox = new MessageBox(shell, SWT.ICON_WARNING | SWT.OK); msgBox.setMessage("To save an empty game list would be like drawing from an empty deck."); @@ -35,6 +34,7 @@ public class HandleSaveGamelist { return; } + // TODO: handle existing name SaveGameListDialog saveDialog = new SaveGameListDialog(shell); saveDialog.open(); diff --git a/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/ui/parts/BatMain.java b/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/ui/parts/BatMain.java index 722f751..cec7827 100644 --- a/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/ui/parts/BatMain.java +++ b/xyz.veronie.bgg.ui/src/xyz/veronie/bgg/ui/parts/BatMain.java @@ -55,6 +55,7 @@ import xyz.veronie.bgg.ui.helpers.BatLayouts; // TODO: Display selection details on game // TODO: cache family/geeklist descriptions // TODO: allow different screen scalings +// TODO: handle existing name @SuppressWarnings("restriction") public class BatMain {