Для библиотеки VCL фирмой Borland реализована собственная версия интерфейса Drag&Drop (переводится как "перетащить"). Интерфейс этот внутренний — передавать и принимать можно любые управляющие элементы Delphi внутри формы' (кроме самой формы). Он реализован без использования соответствующих функций API Windows — их нужно применять при организации общения с другими задачами путем перетаскивания.
Нажав левую кнопку мыши над элементом управления, мы можем "перетащить" его на любой другой элемент. С точки зрения программиста это означает, что в моменты перетаскивания и отпускания клавиши генерируются определенные события, которыми передается вся необходимая информация — указатель на перетаскиваемый объект, текущие координаты курсора и др. Получателем событий является тот элемент, на котором в данный момент находится курсор. Обработчик такого события должен сообщить системе, принимает ли данньш элемент управления "посылку" или нет. При отпускании кнопки над принимающим элементом управления генерируется еще одно или два события, в зависимости от готовности приемника.
Способ работы с этим интерфейсом в VCL определяется свойством:
Code: |
property DragMode: TDragMode; TDragMode = (dmManual, dmAutomatic); |
Для автоматического включения механизмов, имеющихся в VCL, необходимо, чтобы свойство компонента DragMode было установлено в dmAutomatic. Это означает, что на всех стадиях перетаскивания нужные функции вызываются без участия программиста. Его задача состоит только в том, чтобы определить методы-обработчики соответствующих событий. В режиме dmManual (принимаемом по умолчанию) все необходимые вызовы функций нужно делать самому.
Рассмотрим подробнее формат обработчиков трех основных событий интерфейса Drag&Drop.
Во время перетаскивания при перемещении курсора мыши с перетаскиваемым элементом над другим элементом управления, а также при отпускании кнопки для последнего возникает событие:
Code: |
property OnDragOver: TDragOverEvent; TDragOverEvent = procedure(Sender, Source: TObject; X, Y: Integer; State: TDragState; var Accept: Boolean) of object; |
Параметры события:
Sender — элемент, над которьм находится курсор;
Source — перетаскиваемый элемент;
X, Y — координаты курсора (в системе координат клиентской области Sender);
State — одно из трех состояний:
Code: |
TDragState = (dsDragEnter, dsDragLeave, dsDragMove) ; |
Существует одно исключение из этого правила, касающееся компонента TOLEContainer он может принимать объекты OLE m других выполняющихся приложений.
dsDragEnter — курсор мыши появился над элементом;
dsDragMove — курсор мыши переместился над элементом;
dsDragLeave — курсор мыши ушел с элемента или была отпущена кнопка.
Обработчик этого события должен возвратить решение, примет ли данный элемент объект Source или нет, в булевой переменной Accept. Если обработчик этого события отсутствует, то элемент управления не может работать приемником, т. е. на него нельзя "перетащить".
Во время перетаскивания над элементом управления он может изменить вид курсора, сигнализируя о готовности приема. Предназначенный для этого случая курсор описывается свойством:
property DragCursor: TCursor;
Если компонент не является приемником Drag&Drop, или обработчик события OnDragOver отсутствует или возвращает False, то появляется другой курсор (по умолчанию crNoDrop).
Событие:
Code: |
property OnDragDrop: TDragDropEvent ; TDragDropEvent = procedure(Sender, Source: TObject; X, Y: Integer) of object; |
возникает во время перетаскивания при отпускании левой кнопки мыши над элементом, готовым к приему. Параметры его имеют тот же смысл, что и в предыдущем случае.
В приведенном примере вы можете перетащить одну из геометрических фигур в вашей форме (компонент TShape) на статический текст Label 1. Для этого в Label 1 описано два метода: Label IDragOver сигнализирует о готовности принять только компоненты класса TShape, a Label IDragDrop вычисляет площадь фигуры в квадратных сантиметрах в зависимости от ее вида (эллипс, прямоугольник, прямоугольник с закругленными углами).
Code: |
procedure TPormI -LabellDragOver (Sender, Source: TObject; X, Y: Integer; State: TDragState; var Accept: Boolean); begin Accept := Source is TShape; end;
procedure TFormI .LabellDragDrop (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); Labell-Caption := FloatToStrF(f, ffFixed, 5, 2)+ ' кв.см'; end; |
При завышении перетаскивания, вне зависимости от готовности приемника всегда возникает еще одно событие:
Code: |
property OnEndDrag: TEndDragEvent; TEndDragEvent =procedure(Sender, Target: TObject; X, Y: Integer) of object; |
Его параметры идентичны описанным выше.
Для управления перетаскиванием вручную (в режиме dmManual) есть следующие возможности. Начало перетаскивания происходит при вызове метода:
Code: |
procedure BeginDrag(Immediate: Boolean); |
Программист должен связать вызов этого метода с каким-либо событием в системе. (Если свойство DragMode установлено в dmAutomatic, BeginDrag вызывается функцией окна при нажатии левой кнопки мыши). Параметр Immediate определяет, когда именно возникает состояние Drag&Drop: в случае True немедленно, в случае False — после смещения мыши с нажатой левой кнопкой на 5 точек по любой из осей. Последний вариант дает возможность использовать нажатие левой кнопки и для перетаскивания, и для регистрации щелчков на элементе управления (скажем, на кнопке). В режиме dmAutomatic такой возможности нет. Метод:
Code: |
procedure DragDrop(DragObject: TObject; X, Y: Integer); dynamic; |
вызывает обработчик события OnDragDrop, а производит все завершающие действия метод
Code: |
procedure EndDrag(Drop: Boolean); |
Он инициирует события OnDragDrop (при возможности приема) и OnEndDrag.
Метод
Code: |
function Dragging: Boolean; |
возвращает True, если данный элемент в настоящий момент перетаскивается.
Просьба писать ваши замечания, наблюдения и все остальное,
что поможет улучшить предоставляемую информацию на этом сайте.
ВСЕ КОММЕНТАРИИ МОДЕРИРУЮТСЯ ВРУЧНУЮ, ТАК ЧТО СПАМИТЬ БЕСПОЛЕЗНО!