Skip to main content
summaryrefslogblamecommitdiffstats
blob: 6630f3782e9055594d1cff9c622365c7f032b4ad (plain) (tree)
1
2
3
4
5
6
7
8
9
10









                                                                                 
                                                          


                      











                                                                       

                                        



                                                                               







                                                                                
   


                                                                        





                                                                                                





                                                                                                                                  

                                                                                      
                                                                                                                                                                  



                          





                                                                                   



                                                                               
                                                
                                                
                                                                                                              






                                                                                                                                                                    
                                                  










                                                                                                 
                                                                                                                              






                                                                  
                                                    











                                                                                           
                                                                     



                         
                                                








                                                  






                                                                               





                                                                                                                           

















                                                                                                            

                                                                                



















                                                                                     
 
/*******************************************************************************
 * Copyright (c) 2007 Wind River Systems and others.
 * All rights reserved. This program and the accompanying materials
 * are made available under the terms of the Eclipse Public License v1.0
 * which accompanies this distribution, and is available at
 * http://www.eclipse.org/legal/epl-v10.html
 * 
 * Contributors:
 *     Wind River Systems - initial API and implementation
 *******************************************************************************/
package org.eclipse.cdt.dsf.debug.ui.viewmodel.expression;

import java.util.List;

import org.eclipse.cdt.dsf.concurrent.ConfinedToDsfExecutor;
import org.eclipse.cdt.dsf.concurrent.DataRequestMonitor;
import org.eclipse.cdt.dsf.concurrent.IDsfStatusConstants;
import org.eclipse.cdt.dsf.concurrent.MultiRequestMonitor;
import org.eclipse.cdt.dsf.datamodel.IDMContext;
import org.eclipse.cdt.dsf.internal.ui.DsfUIPlugin;
import org.eclipse.cdt.dsf.service.DsfSession;
import org.eclipse.cdt.dsf.ui.concurrent.ViewerDataRequestMonitor;
import org.eclipse.cdt.dsf.ui.viewmodel.IVMContext;
import org.eclipse.cdt.dsf.ui.viewmodel.VMChildrenUpdate;
import org.eclipse.cdt.dsf.ui.viewmodel.datamodel.AbstractDMVMNode;
import org.eclipse.cdt.dsf.ui.viewmodel.datamodel.AbstractDMVMProvider;
import org.eclipse.core.runtime.IStatus;
import org.eclipse.core.runtime.Status;
import org.eclipse.debug.core.model.IExpression;
import org.eclipse.debug.internal.ui.viewers.model.provisional.IChildrenUpdate;

/**
 * Base class for VM Nodes which can be used in the expressions view.  
 * <p>
 * This base class uses the methods {@link #canParseExpression(IExpression)} and
 * {@link #update(IChildrenUpdate[])} to implement the 
 * {@link IExpressionVMNode#update(IExpressionUpdate)}
 * method.  Two additional abstract protected methods need to be implemented
 * by the sub-class as well.   
 * </p>
 */
public abstract class AbstractExpressionVMNode extends AbstractDMVMNode 
    implements IExpressionVMNode
{
    /**
     * @since 2.0
     */    
    protected static final String PROP_ELEMENT_EXPRESSION = "element_expression";  //$NON-NLS-1$

    
    public AbstractExpressionVMNode(AbstractDMVMProvider provider, DsfSession session, Class<? extends IDMContext> dmcClassType) {
        super(provider, session, dmcClassType);
    }

    public void update(final IExpressionUpdate update) {
        if (!canParseExpression(update.getExpression())) {
            // This method should not be called if canParseExpression() returns false.
            // Return an internal error status.
            update.setStatus(new Status(IStatus.ERROR, DsfUIPlugin.PLUGIN_ID, IDsfStatusConstants.INTERNAL_ERROR, "Cannot parse expression", null)); //$NON-NLS-1$
            update.done();
            return;
        }
        
        // Retrieve the list of all elements from the sub-class.  Then compare 
        // each returned element to the expression in the update, using 
        // testElementForExpression().  The element that matches the expression
        // is returned to the client.
        // If no matching element is found, the createInvalidExpressionVMContext() 
        // method is called to a special context.
        update(new IChildrenUpdate[] { new VMChildrenUpdate(
            update, -1, -1,
            new ViewerDataRequestMonitor<List<Object>>(getExecutor(), update) {
                @Override
                protected void handleSuccess() {
                    if (getData().size() == 0) {
                        update.setExpressionElement(createInvalidExpressionVMContext(update.getExpression()));
                        update.done();
                    } else {
                        final List<Object> elements = getData();

                        final MultiRequestMonitor<DataRequestMonitor<Boolean>> multiRm = new MultiRequestMonitor<DataRequestMonitor<Boolean>>(getExecutor(), null) {
                            @Override
                            protected void handleCompleted() {
                                if (isSuccess()) {
                                    boolean foundMatchingContext = false;
                                    for (int i = 0; i < getRequestMonitors().size(); i++) {
                                        if (getRequestMonitors().get(i).getData()) {
                                            Object element = elements.get(i);
                                            associateExpression(element, update.getExpression());
                                            update.setExpressionElement(element);
                                            foundMatchingContext = true;
                                            break;
                                        }
                                    }
                                    if (!foundMatchingContext) {
                                        update.setExpressionElement(createInvalidExpressionVMContext(update.getExpression()));
                                    }
                                } else {
                                    update.setStatus(getStatus());
                                }
                                update.done();
                            }
                        }; 
                        multiRm.requireDoneAdding();
                            
                        for (Object element : elements) {
                            testElementForExpression(
                                element, update.getExpression(), 
                                multiRm.add(
                                    new DataRequestMonitor<Boolean>(getExecutor(), null) { 
                                        @Override
                                        protected void handleCompleted() {
                                            multiRm.requestMonitorDone(this);
                                        }
                                    }));
                        }
                        multiRm.doneAdding();                        
                    }
                }
                
                @Override
                protected void handleFailure() {
                    update.setStatus(getStatus());
                    update.done();
                }
            })}
        );

    }
    
    
    /**
     * Tests whether the given element matches the given expression.
     * 
     * @param element Element to test against the given expression.
     * @param expression Expression to use to check if the element is matching.
     * @param rm The request monitor for the result. 
     */
    @ConfinedToDsfExecutor("#getSession#getExecutor")
    protected void testElementForExpression(Object element, IExpression expression, final DataRequestMonitor<Boolean> rm) {
        rm.setData(false);
        rm.done();
    }
    
    /**
     * Sets the given expression as the expression belonging to the given 
     * element.
     * <p>
     * This base class creates VM context elements using the extending class's 
     * {@link #update(IChildrenUpdate[])} method.  The element matching the 
     * expression is found using {@link #testElementForExpression(Object, IExpression, DataRequestMonitor)}.
     * Once the matching element is found it needs to be linked to the expression
     * so that it can be distinguished from other contexts created for identical
     * but distinct expressions.  This method accomplishes this task.  Elements
     * which are associated with expressions should use the expression object
     * for implementation of {@link #equals(Object)} and {@link #hashCode()}
     * methods. 
     * </p> 
     *  
     * @param element
     * @param expression
     */
    protected void associateExpression(Object element, IExpression expression) {
    }
     
    /**
     * Create a place holder for an invalid expression.  If for a given expression, 
     * this VM node returns true from {@link #canParseExpression(IExpression)}, which
     * indicates that the expression matches the node's expected format, but the node
     * then is not able to find the element represented by the expression, then an 
     * "invalid" expression context needs to be created.  
     * <p>
     * This method can be overriden to provide a node-specific invalid expression 
     * context.
     * </p>
     *
     * @param expression Expression to create the context for.
     * @return Returns a VM context object representing an invalid expression with 
     * 
     * @since 1.1
     */
    protected IVMContext createInvalidExpressionVMContext(IExpression expression) {
        return new InvalidExpressionVMContext(this, expression);
    }
}

Back to the top