package eu.siacs.conversations.entities;

import android.content.Context;
import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import com.google.common.base.Strings;
import com.google.common.collect.ImmutableList;
import eu.siacs.conversations.utils.StringUtils;
import eu.siacs.conversations.utils.UIHelper;
import eu.siacs.conversations.xml.Element;
import eu.siacs.conversations.xmpp.Jid;
import im.conversations.android.xmpp.model.bookmark.Storage;
import im.conversations.android.xmpp.model.bookmark2.Conference;
import im.conversations.android.xmpp.model.bookmark2.Extensions;
import java.lang.ref.WeakReference;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Locale;
import java.util.Map;

public class Bookmark extends Element implements ListItem {

    private final Account account;
    private WeakReference<Conversation> conversation;
    private Jid jid;
    protected Extensions extensions = new Extensions();

    public Bookmark(final Account account, final Jid jid) {
        super("conference");
        this.jid = jid;
        this.setAttribute("jid", jid);
        this.account = account;
    }

    private Bookmark(Account account) {
        super("conference");
        this.account = account;
    }

    public static Map<Jid, Bookmark> parseFromStorage(
            final Storage storage, final Account account) {
        // TODO refactor to use extensions. get rid of the 'old' handling
        if (storage == null) {
            return Collections.emptyMap();
        }
        final HashMap<Jid, Bookmark> bookmarks = new HashMap<>();
        for (final Element item : storage.getChildren()) {
            if (item.getName().equals("conference")) {
                final Bookmark bookmark = Bookmark.parse(item, account);
                if (bookmark != null) {
                    final Bookmark old = bookmarks.put(bookmark.jid, bookmark);
                    if (old != null
                            && old.getBookmarkName() != null
                            && bookmark.getBookmarkName() == null) {
                        bookmark.setBookmarkName(old.getBookmarkName());
                    }
                }
            }
        }
        return bookmarks;
    }

    public static Bookmark parse(Element element, Account account) {
        Bookmark bookmark = new Bookmark(account);
        bookmark.setAttributes(element.getAttributes());
        bookmark.setChildren(element.getChildren());
        bookmark.jid = Jid.Invalid.getNullForInvalid(bookmark.getAttributeAsJid("jid"));
        if (bookmark.jid == null) {
            return null;
        }
        return bookmark;
    }

    public static Bookmark parseFromItem(
            final String id, final Conference conference, final Account account) {
        if (id == null || conference == null) {
            return null;
        }
        final Bookmark bookmark = new Bookmark(account);
        bookmark.jid = Jid.Invalid.getNullForInvalid(Jid.ofOrInvalid(id));
        // TODO verify that we only use bare jids and ignore full jids
        if (bookmark.jid == null) {
            return null;
        }

        // TODO use proper API

        bookmark.setBookmarkName(conference.getAttribute("name"));
        bookmark.setAutojoin(conference.getAttributeAsBoolean("autojoin"));
        bookmark.setNick(conference.findChildContent("nick"));
        bookmark.setPassword(conference.findChildContent("password"));
        final var extensions = conference.getExtensions();
        if (extensions != null) {
            bookmark.extensions = conference.getExtensions();
        }
        return bookmark;
    }

    public Extensions getExtensions() {
        return extensions;
    }

    public void setAutojoin(boolean autojoin) {
        if (autojoin) {
            this.setAttribute("autojoin", "true");
        } else {
            this.setAttribute("autojoin", "false");
        }
    }

    @Override
    public int compareTo(final @NonNull ListItem another) {
        return this.getDisplayName().compareToIgnoreCase(another.getDisplayName());
    }

    @Override
    public String getDisplayName() {
        final Conversation c = getConversation();
        final String name = getBookmarkName();
        if (c != null) {
            return c.getName().toString();
        } else if (printableValue(name, false)) {
            return name.trim();
        } else {
            Jid jid = this.getJid();
            return jid != null && jid.getLocal() != null ? jid.getLocal() : "";
        }
    }

    public static boolean printableValue(@Nullable String value, boolean permitNone) {
        return value != null && !value.trim().isEmpty() && (permitNone || !"None".equals(value));
    }

    public static boolean printableValue(@Nullable String value) {
        return printableValue(value, true);
    }

    @Override
    public Jid getJid() {
        return this.jid;
    }

    public Jid getFullJid() {
        final String nick = Strings.nullToEmpty(getNick()).trim();
        if (jid == null || nick.isEmpty()) {
            return jid;
        }
        try {
            return jid.withResource(nick);
        } catch (final IllegalArgumentException e) {
            return jid;
        }
    }

    @Override
    public List<Tag> getTags(final Context context) {
        final ImmutableList.Builder<Tag> tags = new ImmutableList.Builder<>();
        for (final Element element : getChildren()) {
            final String content = element.getContent();
            if (Strings.isNullOrEmpty(content)) {
                continue;
            }
            if (element.getName().equals("group")) {
                tags.add(new Tag(content));
            }
        }
        return tags.build();
    }

    public String getNick() {
        return Strings.emptyToNull(this.findChildContent("nick"));
    }

    public void setNick(String nick) {
        Element element = this.findChild("nick");
        if (element == null) {
            element = this.addChild("nick");
        }
        element.setContent(nick);
    }

    public boolean autojoin() {
        return this.getAttributeAsBoolean("autojoin");
    }

    public String getPassword() {
        return this.findChildContent("password");
    }

    public void setPassword(String password) {
        Element element = this.findChild("password");
        if (element != null) {
            element.setContent(password);
        }
    }

    @Override
    public boolean match(Context context, String needle) {
        if (needle == null) {
            return true;
        }
        needle = needle.toLowerCase(Locale.US);
        final Jid jid = getJid();
        return (jid != null && jid.toString().contains(needle))
                || getDisplayName().toLowerCase(Locale.US).contains(needle)
                || matchInTag(context, needle);
    }

    private boolean matchInTag(Context context, String needle) {
        needle = needle.toLowerCase(Locale.US);
        for (Tag tag : getTags(context)) {
            if (tag.getName().toLowerCase(Locale.US).contains(needle)) {
                return true;
            }
        }
        return false;
    }

    public Account getAccount() {
        return this.account;
    }

    public synchronized Conversation getConversation() {
        return this.conversation != null ? this.conversation.get() : null;
    }

    public synchronized void setConversation(Conversation conversation) {
        if (this.conversation != null) {
            this.conversation.clear();
        }
        if (conversation == null) {
            this.conversation = null;
        } else {
            this.conversation = new WeakReference<>(conversation);
        }
    }

    public String getBookmarkName() {
        return this.getAttribute("name");
    }

    public boolean setBookmarkName(String name) {
        String before = getBookmarkName();
        if (name != null) {
            this.setAttribute("name", name);
        } else {
            this.removeAttribute("name");
        }
        return StringUtils.changed(before, name);
    }

    @Override
    public int getAvatarBackgroundColor() {
        return UIHelper.getColorForName(
                jid != null ? jid.asBareJid().toString() : getDisplayName());
    }

    @Override
    public String getAvatarName() {
        return getDisplayName();
    }
}
