Справочник по компонентам Delphi. Часть 1 - Внутренний интерфейс Drag&Drop

ОГЛАВЛЕНИЕ

 

Внутренний интерфейс Drag&Drop

Для библиотеки VCL фирмой Borland реализована собственная версия интер­фейса Drag&Drop (переводится как "перетащить"). Интерфейс этот внут­ренний — передавать и принимать можно любые управляющие элементы Del­phi внутри формы' (кроме самой формы). Он реализован без использования соответствующих функций API Windows — их нужно применять при организации общения с другими задачами путем перетаскивания.

Нажав левую кнопку мыши над элементом управления, мы можем "пере­тащить" его на любой другой элемент. С точки зрения программиста это означает, что в моменты перетаскивания и отпускания клавиши генерируются определенные события, которыми передается вся необходимая информация — указатель на перетаскиваемый объект, текущие координаты курсора и др. Получателем событий является тот элемент, на котором в данный момент находится курсор. Обработчик такого события должен сообщить системе, принимает ли данньш элемент управления "посылку" или нет. При отпускании кнопки над принимающим элементом управления генерируется еще одно или два события, в зависимости от готовности приемника.

Способ работы с этим интерфейсом в VCL определяется свойством:

 (Pb) property DragMode: TDragMode;
TDragMode = (dmManual, dmAutomatic);

Для автоматического включения механизмов, имеющихся в VCL, необходимо, чтобы свойство компонента DragMode было установлено в dmAutomatic. Это означает, что на всех стадиях перетаскивания нужные функции вызываются без участия программиста. Его задача состоит только в том, чтобы определить методы-обработчики соответствующих событий. В режиме dmManual (принимаемом по умолчанию) все необходимые вызовы функций нужно делать самому.

Рассмотрим подробнее формат обработчиков трех основных событий интер­фейса Drag&Drop.

Во время перетаскивания при перемещении курсора мыши с перетаскиваемым элементом над другим элементом управления, а также при отпускании кнопки для последнего возникает событие:

(pVl property OnDragOver: TDragOverEvent;
TDragOverEvent = procedure(Sender, Source: TObject; X, Y: Integer;
State: TDragState; var Accept: Boolean) of object; Параметры события:
  • Sender — элемент, над которьм находится курсор;
  • Source — перетаскиваемый элемент;
  • X, Y — координаты курсора (в системе координат клиентской области Sender);
  • State — одно из трех состояний:
TDragState = (dsDragEnter, dsDragLeave, dsDragMove); 

Существует одно исключение из этого правила, касающееся компонента TOLEContainer он может принимать объекты OLE m других выполняющихся приложений.

  • dsDragEnter — курсор мыши появился над элементом;
  • dsDragMove — курсор мыши переместился над элементом;
  • dsDragLeave — курсор мыши ушел с элемента или была отпущена кнопка.

Обработчик этого события должен возвратить решение, примет ли данный элемент объект Source или нет, в булевой переменной Accept. Если обработчик этого события отсутствует, то элемент управления не может работать приемником, т. е. на него нельзя "перетащить".

Во время перетаскивания над элементом управления он может изменить вид курсора, сигнализируя о готовности приема. Предназначенный для этого случая курсор описывается свойством:

 (Pb) property DragCursor: TCursor; 

Если компонент не является приемником Drag&Drop, или обработчик события OnDragOver отсутствует или возвращает False, то появляется другой курсор (по умолчанию crNoDrop).

Событие:

(Pb) property OnDragDrop: TDragDropEvent;
TDragDropEvent = procedure(Sender, Source: TObject; X, Y: Integer) of object;

возникает во время перетаскивания при отпускании левой кнопки мыши над элементом, готовым к приему. Параметры его имеют тот же смысл, что и в предыдущем случае.

В приведенном примере вы можете перетащить одну из геометрических фигур в вашей форме (компонент TShape) на статический текст Label 1. Для этого в Label 1 описано два метода: Label IDragOver сигнализирует о готовности принять только компоненты класса TShape, a Label IDragDrop вычисляет пло­щадь фигуры в квадратных сантиметрах в зависимости от ее вида (эллипс, прямоугольник, прямоугольник с закругленными углами).

procedure TForm1.Label1DragOver (Sender, Source: TObject; X, Y: Integer; State: TDragState; var Accept: Boolean); 
begin
    Accept := Source is TShape;
end;

procedure TForm1.Label1DragDrop (Sender, Source: TObject; X, Y: Integer);
var
    f : single;
    i : Integer;
begin
    with Source as TShape do
    begin
        i := Width;
        if i > Height then
            i:= Height;
        case Shape of
            stRectangle: f := Width*Height;
            stSquare: f := i * i;
            stCircle: f := Pi * i*i / 4;
            stEllipse: f := Pi * Widths-Height / 4;
            stRoundRect, stRoundSquare: begin
                    if Shape = stRoundRect then
                        f := Width*Height
                    else
                        f := i * i;
                    i := (i - Pen.Width + 1) div 4;
                    f := f - (4-Pi)*i*i;
                end;
            end;
    end;
    f := f / Sqr(Forml.PixelsPerInch / 2.54);
    Label1.Caption := FloatToStrF(f, ffFixed, 5, 2)+ ' кв.см';
end;

При завышении перетаскивания, вне зависимости от готовности приемника всегда возникает еще одно событие:

 (Pb) property OnEndDrag: TEndDragEvent;
TEndDragEvent =procedure(Sender, Target: TObject; X, Y: Integer) of object;

Его параметры идентичны описанным выше.

Для управления перетаскиванием вручную (в режиме dmManual) есть следу­ющие возможности. Начало перетаскивания происходит при вызове метода:

 procedure BeginDrag(Immediate: Boolean); 

Программист должен связать вызов этого метода с каким-либо событием в системе. (Если свойство DragMode установлено в dmAutomatic, BeginDrag вы­зывается функцией окна при нажатии левой кнопки мыши). Параметр Imme­diate определяет, когда именно возникает состояние Drag&Drop: в случае True немедленно, в случае False — после смещения мыши с нажатой левой кнопкой на 5 точек по любой из осей. Последний вариант дает возможность исполь­зовать нажатие левой кнопки и для перетаскивания, и для регистрации щелчков на элементе управления (скажем, на кнопке). В режиме dmAutomatic такой возможности нет. Метод:

 procedure DragDrop(DragObject: TObject; X, Y: Integer); dynamic; 

вызывает обработчик события OnDragDrop, а производит все завершающие действия метод

procedure EndDrag(Drop: Boolean); 

Он инициирует события OnDragDrop (при возможности приема) и OnEndDrag.

 Метод function Dragging: Boolean; 

возвращает True, если данный элемент в настоящий момент перетаскивается.