以编程方式滚动 ScrolledComposite TableViewer 项目

programmatically scroll ScrolledComposite TableViewer item

我在 ScrolledComposite 上有一个带有大型 table 查看器的对话框。 我需要以编程方式将 ScrolledComposite 滚动到 TableViewer 中的 select 项。 看起来像是一项简单的任务,但我真的得到了堆栈。 我尝试了很多想法,但没有一个有效。

有我的示例代码:

import org.eclipse.jface.viewers.ArrayContentProvider;
import org.eclipse.jface.viewers.LabelProvider;
import org.eclipse.jface.viewers.TableViewer;
import org.eclipse.swt.SWT;
import org.eclipse.swt.custom.ScrolledComposite;
import org.eclipse.swt.layout.FillLayout;
import org.eclipse.swt.layout.GridData;
import org.eclipse.swt.layout.GridLayout;
import org.eclipse.swt.widgets.Composite;
import org.eclipse.swt.widgets.Display;
import org.eclipse.swt.widgets.Shell;
import org.eclipse.swt.widgets.TableItem;

/**
* Scroll a Viewer 99th element
*
*/
public class Snippet008RevealElement {

public class MyModel {
    public int counter;

    public MyModel(int counter) {
        this.counter = counter;
    }

    @Override
    public String toString() {
        return "Item " + this.counter;
    }
}

public Snippet008RevealElement(Shell shell) {
    ScrolledComposite scrolledComposite = new ScrolledComposite(shell, SWT.V_SCROLL | SWT.H_SCROLL);

    GridData data = new GridData(SWT.FILL, SWT.FILL, true, true);
    scrolledComposite.setLayoutData(data);

    scrolledComposite.setExpandHorizontal(true);
    scrolledComposite.setExpandVertical(true);

    Composite main = new Composite(scrolledComposite, SWT.NONE);
    main.setLayout(new GridLayout());
    main.setLayoutData(new GridData(GridData.FILL_BOTH));


    final TableViewer v = new TableViewer(main);
    v.setLabelProvider(new LabelProvider());
    v.setContentProvider(ArrayContentProvider.getInstance());
    MyModel[] model = createModel();
    v.setInput(model);
    v.getTable().setLinesVisible(true);

 //     v.reveal(model[99]);
 //     v.getTable().setSelection(99);

    TableItem[] items = v.getTable().getItems();
    TableItem item = items[99];
    scrolledComposite.getVerticalBar().setSelection(item.getBounds().y);

    scrolledComposite.setContent(main);
    scrolledComposite.setMinSize(main.computeSize(SWT.DEFAULT, SWT.DEFAULT));
}

private MyModel[] createModel() {
    MyModel[] elements = new MyModel[100];

    for( int i = 0; i < 100; i++ ) {
        elements[i] = new MyModel(i);
    }

    return elements;
}

/**
 * @param args
 */
public static void main(String[] args) {
    Display display = new Display ();
    Shell shell = new Shell(display);
    shell.setLayout(new FillLayout());
    new Snippet008RevealElement(shell);
    shell.open ();

    while (!shell.isDisposed ()) {
        if (!display.readAndDispatch ()) display.sleep ();
    }

    display.dispose ();

}

}

真正的挑战或错误是为 TableItem item = items[99] 找到真正的界限;在复合材料上不可见。

您可以使用 ScrolledCompositesetOrigin 方法,例如:

scrolledComposite.setContent(main);
scrolledComposite.setMinSize(main.computeSize(SWT.DEFAULT, SWT.DEFAULT));

// Need to run the calculations after all size calculations have been done
// So do asynchronously. 

final Display display = scrolledComposite.getDisplay();

display.asyncExec(() ->
{
  final Table table = v.getTable();

  final TableItem item = table.getItem(99);

  Rectangle itemBounds = item.getBounds();

  // Convert to be relative to scrolled composite

  itemBounds = display.map(viewer.getTable(), scrolledComposite, itemBounds);

  scrolledComposite.setOrigin(0, itemBounds.y);
 });

注意:如果您在控件初始化期间调用此代码,则边界计算不准确,因此我在这里展示了它是异步完成的。 asyncExec 如果您 运行 来自按钮或类似代码的代码,则不需要 asyncExec