X-Git-Url: http://git.nikiroo.be/?a=blobdiff_plain;f=src%2Fbe%2Fnikiroo%2Fjvcard%2Ftui%2Fpanes%2FContactDetailsRaw.java;h=c29ba00ec36a099d5a3f118bd88e41c39aba4fca;hb=8002675fff4eaeeb292c4b1a1e4182bd999a548c;hp=271be442e82922e1ef30c0bd7dad61ee1ab3d5b1;hpb=f04d8b1c4c3ed29d4d23cc076f307ef455b2dcb6;p=jvcard.git diff --git a/src/be/nikiroo/jvcard/tui/panes/ContactDetailsRaw.java b/src/be/nikiroo/jvcard/tui/panes/ContactDetailsRaw.java index 271be44..c29ba00 100644 --- a/src/be/nikiroo/jvcard/tui/panes/ContactDetailsRaw.java +++ b/src/be/nikiroo/jvcard/tui/panes/ContactDetailsRaw.java @@ -3,50 +3,341 @@ package be.nikiroo.jvcard.tui.panes; 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 be.nikiroo.jvcard.tui.TuiLauncher; + +import com.googlecode.lanterna.gui2.MultiWindowTextGUI; +import com.googlecode.lanterna.gui2.dialogs.ActionListDialogBuilder; +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.size(); i++) { + addItem("x"); + } + } + + @Override + public DataType getDataType() { + return DataType.DATA; + } + + @Override + public List getKeyBindings() { + List actions = new LinkedList(); + + 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 and i18n + actions.add(new KeyAction(Mode.ASK_USER, 'a', StringId.KEY_ACTION_ADD) { + private String name; + + @Override + public boolean onAction() { + new ActionListDialogBuilder() + .setTitle("New data") + .setDescription("Choose a data type") + .addAction("Email address", new Runnable() { + @Override + public void run() { + name = "EMAIL"; + } + }) + .addAction("Telephone number", new Runnable() { + @Override + public void run() { + name = "TEL"; + } + }) + .addAction("Birthday", new Runnable() { + @Override + public void run() { + name = "BDAY"; + } + }) + .addAction("[other]", new Runnable() { + @Override + public void run() { + name = ""; + } + }) + .build() + .showDialog( + new MultiWindowTextGUI(TuiLauncher.getScreen())); + + String name = this.name; + this.name = null; + + if (name != null) { + if (name.length() > 0) { + Data data = new Data(null, name, "", null); + getContact().add(data); + addItem("x"); + } + + return (name.length() == 0); + } + + return false; + } + + @Override + public Object getObject() { + return contact; + } + + @Override + public String getQuestion() { + // TODO i18n + if (name == null) + return "Data name: "; + + return null; + } + + @Override + public String callback(String answer) { + if (answer != null & answer.length() > 0) { + Data data = new Data(null, answer, "", null); + 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(); + } - for (int i = 0; i < contact.getContent().size(); i++) { - addItem("[detail line]"); + @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 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 parts = new LinkedList(); + 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)); @@ -55,93 +346,145 @@ public class ContactDetailsRaw extends MainContentList { } 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(""); + } else { + valueBuilder.append(data.getValue().replaceAll("\n", "\\\\n")); + } if (data.getGroup() != null && data.getGroup().length() > 0) { - valueBuilder.append("("); + 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; + /** + * 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 getKeyBindings() { - // TODO Auto-generated method stub - List actions = new LinkedList(); - // 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; + /** + * 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(); - return false; - } - }); + for (TypeInfo type : data) { + if (builder.length() > 1) + builder.append(", "); + builder.append(type.getName().replaceAll(",", "\\,")); + builder.append(": "); + builder.append(type.getValue().replaceAll(":", "\\:")); + } - return actions; + return builder; } - @Override - public String getTitle() { - String title = null; + /** + * Unserialise a list of {@link TypeInfo}s. + * + * @param value + * the serialised value + * + * @return the {@link TypeInfo} in their object form + */ + static private List stringToTypes(String value) { + List infos = new LinkedList(); + if (value == null || value.length() == 0) + return infos; - if (contact != null) { - title = contact.getPreferredDataValue("FN"); - if (title == null || title.length() == 0) - title = contact.getPreferredDataValue("N"); - } + 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 title; - } + 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 String move(int x, int y) { - // TODO Auto-generated method stub - return null; + String n = ""; + String v = ""; + if (indexColumn >= 0) { + n = subValue.substring(0, indexColumn).trim(); + v = subValue.substring(indexColumn + 1) + .replaceAll("\\\\:", ":").trim(); + } else { + n = subValue.trim(); + } + + infos.add(new TypeInfo(n, v)); + + done = index + 1; + } + + previous = car; + } + + return infos; } }