import java.util.LinkedList;
import java.util.List;
-import com.googlecode.lanterna.input.KeyType;
-
import be.nikiroo.jvcard.Contact;
import be.nikiroo.jvcard.Data;
import be.nikiroo.jvcard.TypeInfo;
-import be.nikiroo.jvcard.i18n.Trans;
+import be.nikiroo.jvcard.launcher.Main;
+import be.nikiroo.jvcard.resources.StringUtils;
+import be.nikiroo.jvcard.resources.enums.ColorOption;
+import be.nikiroo.jvcard.resources.enums.StringId;
import be.nikiroo.jvcard.tui.KeyAction;
-import be.nikiroo.jvcard.tui.UiColors;
import be.nikiroo.jvcard.tui.KeyAction.DataType;
import be.nikiroo.jvcard.tui.KeyAction.Mode;
-import be.nikiroo.jvcard.tui.StringUtils;
-import be.nikiroo.jvcard.tui.UiColors.Element;
+
+import com.googlecode.lanterna.input.KeyType;
public class ContactDetailsRaw extends MainContentList {
private Contact contact;
- private int mode;
+ private boolean extMode;
public ContactDetailsRaw(Contact contact) {
- super(null, null);
-
this.contact = contact;
- this.mode = 0;
+ this.extMode = false;
- for (int i = 0; i < contact.getContent().size(); i++) {
- addItem("[detail line]");
+ for (int i = 0; i < contact.size(); i++) {
+ addItem("x");
}
}
+ @Override
+ public DataType getDataType() {
+ return DataType.DATA;
+ }
+
+ @Override
+ public List<KeyAction> getKeyBindings() {
+ List<KeyAction> actions = new LinkedList<KeyAction>();
+
+ actions.add(new KeyAction(Mode.ASK_USER, KeyType.Enter,
+ StringId.KEY_ACTION_EDIT_FIELD) {
+ @Override
+ public Object getObject() {
+ Data data = getSelectedData();
+ if (data != null && data.getB64Key() != -1) {
+ setMessage("Cannot modify binary values in RAW mode", true);
+ data = null;
+ }
+
+ return data;
+ }
+
+ @Override
+ public String getQuestion() {
+ Data data = getData();
+ if (data != null) {
+ return data.getName();
+ }
+
+ return null;
+ }
+
+ @Override
+ public String getDefaultAnswer() {
+ Data data = getData();
+ if (data != null) {
+ return data.getRawValue().replaceAll("\n", "\\\\n");
+ }
+
+ return null;
+ }
+
+ @Override
+ public String callback(String answer) {
+ Data data = getData();
+ if (data != null) {
+ data.setRawValue(answer.replaceAll("\\\\n", "\n"));
+ return null;
+ }
+
+ // TODO: i18n
+ return "Cannot modify value";
+ }
+ });
+ actions.add(new KeyAction(Mode.ASK_USER_KEY, 'd',
+ StringId.KEY_ACTION_DELETE) {
+ @Override
+ public Object getObject() {
+ return getSelectedData();
+ }
+
+ @Override
+ public String getQuestion() {
+ // TODO i18n
+ return "Delete data? [Y/N]";
+ }
+
+ @Override
+ public String callback(String answer) {
+ if (answer.equalsIgnoreCase("y")) {
+ Data data = getData();
+ if (data != null && data.delete()) {
+ removeItem("x");
+ return null;
+ }
+
+ // TODO i18n
+ return "Cannot delete data";
+ }
+
+ return null;
+ }
+ });
+ // TODO: ui
+ actions.add(new KeyAction(Mode.ASK_USER, 'a', StringId.KEY_ACTION_ADD) {
+ @Override
+ public Object getObject() {
+ return contact;
+ }
+
+ @Override
+ public String getQuestion() {
+ // TODO i18n
+ return "new data (xx.group = yy): ";
+ }
+
+ @Override
+ public String callback(String answer) {
+ int indexEq = answer.indexOf('=');
+ if (indexEq >= 0) {
+ String name = answer.substring(0, indexEq).trim();
+ String value = answer.substring(indexEq + 1).trim();
+ String group = null;
+
+ int indexDt = name.indexOf('.');
+ if (indexDt >= 0) {
+ group = name.substring(indexDt + 1).trim();
+ name = name.substring(0, indexDt).trim();
+ }
+
+ Data data = new Data(null, name, value, group);
+ getContact().add(data);
+ addItem("x");
+ }
+ return null;
+ }
+ });
+ // TODO: use a real UI for this, not a simple text box (a list or
+ // something, maybe a whole new pane?)
+ actions.add(new KeyAction(Mode.ASK_USER, 't', StringId.DUMMY) {
+ private String previous;
+
+ @Override
+ public Object getObject() {
+ return getSelectedData();
+ }
+
+ @Override
+ public String getQuestion() {
+ Data data = getData();
+ if (data != null) {
+ return data.getName();
+ }
+
+ return null;
+ }
+
+ @Override
+ public String getDefaultAnswer() {
+ Data data = getData();
+ if (data != null) {
+ previous = typesToString(data, null).toString();
+ return previous;
+ }
+
+ return null;
+ }
+
+ @Override
+ public String callback(String answer) {
+ Data data = getData();
+ if (data != null) {
+ if (!answer.equals(previous)) {
+ data.replaceListContent(stringToTypes(answer));
+ }
+ return null;
+ }
+
+ // TODO: i18n
+ return "Cannot modify value";
+ }
+ });
+ actions.add(new KeyAction(Mode.ASK_USER, 'g', StringId.DUMMY) {
+ private String previous;
+
+ @Override
+ public Object getObject() {
+ return getSelectedData();
+ }
+
+ @Override
+ public String getQuestion() {
+ Data data = getData();
+ if (data != null) {
+ return data.getName();
+ }
+
+ return null;
+ }
+
+ @Override
+ public String getDefaultAnswer() {
+ Data data = getData();
+ if (data != null) {
+ previous = data.getGroup();
+ return previous;
+ }
+
+ return null;
+ }
+
+ @Override
+ public String callback(String answer) {
+ Data data = getData();
+ if (data != null) {
+ if (!answer.equals(previous)) {
+ data.setGroup(answer);
+ }
+ return null;
+ }
+
+ // TODO: i18n
+ return "Cannot modify group";
+ }
+ });
+ actions.add(new KeyAction(Mode.NONE, KeyType.Tab,
+ StringId.KEY_ACTION_SWITCH_FORMAT) {
+ @Override
+ public boolean onAction() {
+ extMode = !extMode;
+ return false;
+ }
+ });
+
+ return actions;
+ }
+
+ @Override
+ public String getTitle() {
+ String title = null;
+
+ if (contact != null) {
+ title = contact.getPreferredDataValue("FN");
+ if (title == null || title.length() == 0)
+ title = contact.getPreferredDataValue("N");
+ }
+
+ return title;
+ }
+
+ @Override
+ public String move(int x, int y) {
+ // TODO Auto-generated method stub
+ return null;
+ }
+
@Override
protected List<TextPart> getLabel(int index, int width, boolean selected,
boolean focused) {
+
// TODO: from ini file?
int SIZE_COL_1 = 15;
+ int SIZE_COL_2_OPT = 10;
- Element el = (focused && selected) ? Element.CONTACT_LINE_SELECTED
- : Element.CONTACT_LINE;
- Element elSep = (focused && selected) ? Element.CONTACT_LINE_SEPARATOR_SELECTED
- : Element.CONTACT_LINE_SEPARATOR;
- Element elDirty = (focused && selected) ? Element.CONTACT_LINE_DIRTY_SELECTED
- : Element.CONTACT_LINE_DIRTY;
-
- Data data = contact.getContent().get(index);
+ if (!extMode)
+ SIZE_COL_2_OPT = 0;
List<TextPart> parts = new LinkedList<TextPart>();
+ Data data = null;
+ if (index > -1 && index < contact.size())
+ data = contact.get(index);
+
+ if (data == null)
+ return parts;
+
+ ColorOption el = (focused && selected) ? ColorOption.CONTACT_LINE_SELECTED
+ : ColorOption.CONTACT_LINE;
+ ColorOption elSep = (focused && selected) ? ColorOption.CONTACT_LINE_SEPARATOR_SELECTED
+ : ColorOption.CONTACT_LINE_SEPARATOR;
+ ColorOption elDirty = (focused && selected) ? ColorOption.CONTACT_LINE_DIRTY_SELECTED
+ : ColorOption.CONTACT_LINE_DIRTY;
+
if (data.isDirty()) {
parts.add(new TextPart(" ", el));
parts.add(new TextPart("*", elDirty));
}
String name = " " + data.getName() + " ";
String value = null;
+ String group = null;
StringBuilder valueBuilder = new StringBuilder(" ");
- switch (mode) {
- case 0:
- valueBuilder.append(data.getValue());
+ if (!extMode) {
+ if (data.getB64Key() != -1) {
+ // TODO: i18n
+ valueBuilder.append("<BKey " + data.getB64Key() + ">");
+ } else {
+ valueBuilder.append(data.getValue().replaceAll("\n", "\\\\n"));
+ }
if (data.getGroup() != null && data.getGroup().length() > 0) {
valueBuilder.append("(");
valueBuilder.append(data.getGroup());
valueBuilder.append(")");
}
- break;
- case 1:
- for (TypeInfo type : data.getTypes()) {
- if (valueBuilder.length() > 1)
- valueBuilder.append(", ");
- valueBuilder.append(type.getName());
- valueBuilder.append(": ");
- valueBuilder.append(type.getValue());
- }
- break;
+ } else {
+ group = data.getGroup();
+ if (group == null)
+ group = "";
+
+ typesToString(data, valueBuilder);
}
valueBuilder.append(" ");
value = valueBuilder.toString();
- name = StringUtils.sanitize(name, UiColors.getInstance().isUnicode());
- value = StringUtils.sanitize(value, UiColors.getInstance().isUnicode());
+ name = StringUtils.sanitize(name, Main.isUnicode());
+ value = StringUtils.sanitize(value, Main.isUnicode());
name = StringUtils.padString(name, SIZE_COL_1);
+ group = StringUtils.padString(group, SIZE_COL_2_OPT);
value = StringUtils.padString(value, width - SIZE_COL_1
- - getSeparator().length() - 2);
+ - SIZE_COL_2_OPT - (extMode ? 2 : 1) * getSeparator().length()
+ - 2);
parts.add(new TextPart(name, el));
parts.add(new TextPart(getSeparator(), elSep));
parts.add(new TextPart(value, el));
+ if (extMode) {
+ parts.add(new TextPart(getSeparator(), elSep));
+ parts.add(new TextPart(group, el));
+ }
return parts;
- };
-
- @Override
- public DataType getDataType() {
- return DataType.DATA;
}
- @Override
- public String getExitWarning() {
- // TODO Auto-generated method stub
+ /**
+ * Return the currently selected {@link Data}.
+ *
+ * @return the currently selected {@link Data}
+ */
+ private Data getSelectedData() {
+ int index = getSelectedIndex();
+ if (index > -1 && index < this.contact.size())
+ return contact.get(index);
return null;
}
- @Override
- public List<KeyAction> getKeyBindings() {
- // TODO Auto-generated method stub
- List<KeyAction> actions = new LinkedList<KeyAction>();
+ /**
+ * Serialise the {@link TypeInfo}s in the given {@link Data}.
+ *
+ * @param data
+ * the {@link Data} from which to take the {@link TypeInfo}s
+ * @param builder
+ * an optional {@link StringBuilder} to append the serialised
+ * version to
+ *
+ * @return the given {@link StringBuilder} or a new one if the given one is
+ * NULL
+ */
+ static private StringBuilder typesToString(Data data, StringBuilder builder) {
+ if (builder == null)
+ builder = new StringBuilder();
- // TODO: add, remove
- actions.add(new KeyAction(Mode.EDIT_DETAIL, 'd', Trans.StringId.DUMMY) {
- @Override
- public Object getObject() {
- return contact.getContent().get(getSelectedIndex());
- }
- });
- actions.add(new KeyAction(Mode.NONE, KeyType.Tab,
- Trans.StringId.KEY_ACTION_SWITCH_FORMAT) {
- @Override
- public boolean onAction() {
- mode++;
- if (mode > 1)
- mode = 0;
+ for (TypeInfo type : data) {
+ if (builder.length() > 1)
+ builder.append(", ");
+ builder.append(type.getName().replaceAll(",", "\\,"));
+ builder.append(": ");
+ builder.append(type.getValue().replaceAll(":", "\\:"));
+ }
- return false;
+ return builder;
+ }
+
+ /**
+ * Unserialise a list of {@link TypeInfo}s.
+ *
+ * @param value
+ * the serialised value
+ *
+ * @return the {@link TypeInfo} in their object form
+ */
+ static private List<TypeInfo> stringToTypes(String value) {
+ List<TypeInfo> infos = new LinkedList<TypeInfo>();
+ if (value == null || value.length() == 0)
+ return infos;
+
+ char previous = '\0';
+ char car = '\0';
+ int done = 0;
+ for (int index = 0; index < value.length(); index++) {
+ car = value.charAt(index);
+ if (index == value.length() - 1) {
+ index++;
+ previous = '\0';
+ car = ',';
}
- });
- return actions;
- }
+ if (previous != '\\' && car == ',') {
+ String subValue = value.substring(done, index);
+ int indexColumn = subValue.indexOf(':');
+ while (indexColumn > 0
+ && subValue.charAt(indexColumn - 1) == '\\') {
+ if (indexColumn == subValue.length() - 1) {
+ indexColumn = -1;
+ } else {
+ indexColumn = subValue.indexOf(':', indexColumn + 1);
+ }
+ }
- @Override
- public Mode getMode() {
- return Mode.CONTACT_DETAILS;
- }
+ String n = "";
+ String v = "";
+ if (indexColumn >= 0) {
+ n = subValue.substring(0, indexColumn).trim();
+ v = subValue.substring(indexColumn + 1)
+ .replaceAll("\\\\:", ":").trim();
+ } else {
+ n = subValue.trim();
+ }
- @Override
- public String getTitle() {
- String title = null;
+ infos.add(new TypeInfo(n, v));
- if (contact != null) {
- title = contact.getPreferredDataValue("FN");
- if (title == null || title.length() == 0)
- title = contact.getPreferredDataValue("N");
- }
+ done = index + 1;
+ }
- return title;
- }
+ previous = car;
+ }
- @Override
- public String move(int x, int y) {
- // TODO Auto-generated method stub
- return null;
+ return infos;
}
}