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 {