Ваадин Дерево.Ленивая загрузка - PullRequest
4 голосов
/ 19 марта 2012

Я пытаюсь реализовать отложенную загрузку элементов на узле Expand Events. Есть проблема с этим. Поэтому при расширении события я создаю новые элементы для расширенного узла, используя

HierarchicalContainer.addItem() 

Но этот метод вызывает

containerItemSetChange()

метод класса AbstractSelect. Там

itemIdMapper.removeAll()

называется. Таким образом, карта очищается, и все элементы дерева получают новые идентификаторы на карте. Когда ответ от сервера приходит на сторону клиента, он не знает этот новый идентификатор, потому что у него есть предыдущий идентификатор (расширенного узла), поэтому новые элементы не добавляются и не отображаются на стороне клиента.

Я пытался https://vaadin.com/forum/-/message_boards/view_message/131802. Но на самом деле данные для всего дерева отправляются с сервера на клиент. И я ожидаю, что отправляются только данные, необходимые для конкретного узла. Так что для этого нет производительности «+».

Может ли кто-нибудь помочь мне с этой проблемой? Заранее спасибо.

Edition

На самом деле мне также нужно было изменить поле

partialUpdate

в true, потому что сначала сервер отправляет все содержимое дерева на сторону клиента.

Ответы [ 2 ]

3 голосов
/ 03 апреля 2012

Перефразируя вашу проблему;большинство людей будет предполагать, что дерево использует ваши хеш-коды / равнозначные реализации для ваших объектов (вот почему они есть!) для сравнения.Вы создаете свое дерево, делаете кучу других вещей, включая изменение элементов, которые вы помещаете в дерево, а затем пытаетесь изменить дерево, используя эти элементы в качестве ключа.

Вот проблема.

Дерево в основном делает снимок хэш-кода для элемента при вызове addItem().Этот хэш-код используется им для сопоставления клиентских артефактов с объектом на сервере.Для любых последующих вызовов removeItem() и т. Д. Он просто использует этот хэш-код моментального снимка вместо вызова метода hashcode / equals объекта, как в любом другом обычном контейнере.Если вы выполните отладку через removeItem(), вы довольно быстро поймете, в чем проблема.

  • Простое решение: не модифицируйте ваши объекты, как только вы поместите их вдерево.Но если вам нужно перестроить все дерево.
  • Лучшее решение: создать прокси-объект (TreeNode или что-то в этом роде), который содержит элемент, но предоставляет свои собственные методы equals / hashcode,
  • Лучшее решение: Ваадин переработал свое дерево так, чтобы оно работало как стандартный контейнер java.В его нынешнем виде это странное поведение не упоминается в javadoc о методах, которые могут дать сбой.

Надеюсь, это кому-нибудь поможет.

1 голос
/ 02 декабря 2016

Хотя в документации Vaadin говорится, что отложенная загрузка для Tree не поддерживается , мне удалось реализовать следующий интерфейс отложенная загрузка Hierarchical.Очень важно хранить все элементы в локальной структуре (в моем случае в HashMap hierarchy), не читать элементы несколько раз, это не работает.Я думаю, потому что Ваадин не использует equals() и hashCode().

import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import com.softmodeler.common.CommonPlugin;
import com.softmodeler.model.OutputNode;
import com.softmodeler.service.IViewService;
import com.vaadin.data.Container.Hierarchical;
import com.vaadin.data.Item;
import com.vaadin.data.Property;
import com.vaadin.data.util.BeanItem;

/**
 * @author Flavio Donzé
 * @version 1.0
 */
public class OutputNodeHierachical implements Hierarchical {
    private static final long serialVersionUID = 8289589835030184018L;

    /** the view service */
    private IViewService service = CommonPlugin.getService(IViewService.class);

    /** collection of all root nodes */
    private List<OutputNode> rootNodes = null;
    /** parent=>children mapping */
    private Map<OutputNode, List<OutputNode>> hierarchy = new HashMap<>();

    /**
     * constructor
     *
     * @param rootNodes collection of all root nodes
     */
    public OutputNodeHierachical(List<OutputNode> rootNodes) {
        this.rootNodes = Collections.unmodifiableList(rootNodes);

        addToHierarchy(rootNodes);
    }

    @Override
    public Collection<?> getChildren(Object itemId) {
        try {
            List<OutputNode> children = hierarchy.get(itemId);
            if (children == null) {
                OutputNode node = (OutputNode) itemId;
                children = service.getChildren(node.getNodeId(), false);

                hierarchy.put(node, children);

                // add children to hierarchy, their children will be added on click
                addToHierarchy(children);
            }
            return children;
        } catch (Exception e) {
            VaadinUtil.handleException(e);
        }
        return null;
    }

    /**
     * add each element to the hierarchy without their children hierarchy(child=>null)
     *
     * @param children elements to add
     */
    private void addToHierarchy(List<OutputNode> children) {
        for (OutputNode child : children) {
            hierarchy.put(child, null);
        }
    }

    @Override
    public boolean areChildrenAllowed(Object itemId) {
        return !((OutputNode) itemId).getChilds().isEmpty();
    }

    @Override
    public boolean hasChildren(Object itemId) {
        return !((OutputNode) itemId).getChilds().isEmpty();
    }

    @Override
    public Object getParent(Object itemId) {
        String parentId = ((OutputNode) itemId).getParentId();
        for (OutputNode node : hierarchy.keySet()) {
            if (node.getNodeId().equals(parentId)) {
                return node;
            }
        }
        return null;
    }

    @Override
    public Collection<?> rootItemIds() {
        return rootNodes;
    }

    @Override
    public boolean isRoot(Object itemId) {
        return rootNodes.contains(itemId);
    }

    @Override
    public Item getItem(Object itemId) {
        return new BeanItem<OutputNode>((OutputNode) itemId);
    }

    @Override
    public boolean containsId(Object itemId) {
        return hierarchy.containsKey(itemId);
    }

    @Override
    public Collection<?> getItemIds() {
        return hierarchy.keySet();
    }

    @Override
    public int size() {
        return hierarchy.size();
    }

    @Override
    public boolean setParent(Object itemId, Object newParentId) throws UnsupportedOperationException {
        throw new UnsupportedOperationException();
    }

    @Override
    public boolean setChildrenAllowed(Object itemId, boolean areChildrenAllowed) throws UnsupportedOperationException {
        throw new UnsupportedOperationException();
    }

    @Override
    public Item addItem(Object itemId) throws UnsupportedOperationException {
        throw new UnsupportedOperationException();
    }

    @Override
    public Object addItem() throws UnsupportedOperationException {
        throw new UnsupportedOperationException();
    }

    @Override
    public boolean removeItem(Object itemId) throws UnsupportedOperationException {
        throw new UnsupportedOperationException();
    }

    @Override
    public boolean removeAllItems() throws UnsupportedOperationException {
        throw new UnsupportedOperationException();
    }

    @Override
    public Class<?> getType(Object propertyId) {
        throw new UnsupportedOperationException();
    }

    @Override
    public Collection<?> getContainerPropertyIds() {
        throw new UnsupportedOperationException();
    }

    @Override
    public Property<?> getContainerProperty(Object itemId, Object propertyId) {
        throw new UnsupportedOperationException();
    }

    @Override
    public boolean addContainerProperty(Object propertyId, Class<?> type, Object defaultValue) throws UnsupportedOperationException {
        throw new UnsupportedOperationException();
    }

    @Override
    public boolean removeContainerProperty(Object propertyId) throws UnsupportedOperationException {
        throw new UnsupportedOperationException();
    }

}

Добавление контейнера в Tree следующим образом:

OutputNodeHierachical dataSource = new OutputNodeHierachical(rootNodes);

Tree mainTree = new Tree();
mainTree.setSizeFull();
mainTree.setContainerDataSource(dataSource);
mainTree.addItemClickListener(new ItemClickListener() {
    private static final long serialVersionUID = -413371711541672605L;

    @Override
    public void itemClick(ItemClickEvent event) {
        OutputNode node = (OutputNode) event.getItemId();
        openObject(node.getObjectId());
    }
});

Надеюсь, этот пример поможет, так как я не нашел реального примера в Интернете.

...