]> gerrit.simantics Code Review - simantics/platform.git/blob - bundles/org.eclipse.swt.win32.win32.x86_64/src/org/eclipse/swt/dnd/OleEnumFORMATETC.java
Work around SWT 4.13 - 4.18 Win32 DnD bug 567422
[simantics/platform.git] / bundles / org.eclipse.swt.win32.win32.x86_64 / src / org / eclipse / swt / dnd / OleEnumFORMATETC.java
1 /*******************************************************************************
2  * Copyright (c) 2000, 2017 IBM Corporation and others.
3  *
4  * This program and the accompanying materials
5  * are made available under the terms of the Eclipse Public License 2.0
6  * which accompanies this distribution, and is available at
7  * https://www.eclipse.org/legal/epl-2.0/
8  *
9  * SPDX-License-Identifier: EPL-2.0
10  *
11  * Contributors:
12  *     IBM Corporation - initial API and implementation
13  *******************************************************************************/
14 package org.eclipse.swt.dnd;
15
16 import org.eclipse.swt.internal.*;
17 import org.eclipse.swt.internal.ole.win32.*;
18 import org.eclipse.swt.internal.win32.*;
19
20 final class OleEnumFORMATETC {
21
22         private COMObject iEnumFORMATETC;
23
24         private int refCount;
25         private int index;
26
27         private FORMATETC[] formats;
28
29 OleEnumFORMATETC() {
30
31         createCOMInterfaces();
32
33 }
34 int AddRef() {
35         refCount++;
36         return refCount;
37 }
38 private void createCOMInterfaces() {
39         iEnumFORMATETC = new COMObject(new int[] {2, 0, 0, 3, 1, 0, 1}){
40                 @Override
41                 public long method0(long[] args) {return QueryInterface(args[0], args[1]);}
42                 @Override
43                 public long method1(long[] args) {return AddRef();}
44                 @Override
45                 public long method2(long[] args) {return Release();}
46                 @Override
47                 public long method3(long[] args) {return Next((int)args[0], args[1], args[2]);}
48                 @Override
49                 public long method4(long[] args) {return Skip((int)args[0]);}
50                 @Override
51                 public long method5(long[] args) {return Reset();}
52                 // method6 Clone - not implemented
53         };
54 }
55 private void disposeCOMInterfaces() {
56         if (iEnumFORMATETC != null)
57                 iEnumFORMATETC.dispose();
58         iEnumFORMATETC = null;
59 }
60 long getAddress() {
61         return iEnumFORMATETC.getAddress();
62 }
63 private FORMATETC[] getNextItems(int numItems){
64
65         if (formats == null || numItems < 1) return null;
66
67         int endIndex = index + numItems - 1;
68         if (endIndex > (formats.length - 1)) endIndex = formats.length - 1;
69         if (index > endIndex) return null;
70
71         FORMATETC[] items =  new FORMATETC[endIndex - index + 1];
72         for (int i = 0; i < items.length; i++){
73                 items[i] = formats[index];
74                 index++;
75         }
76
77         return items;
78 }
79 private int Next(int celt, long rgelt, long pceltFetched) {
80         /* Retrieves the next celt items in the enumeration sequence.
81            If there are fewer than the requested number of elements left in the sequence,
82            it retrieves the remaining elements.
83            The number of elements actually retrieved is returned through pceltFetched
84            (unless the caller passed in NULL for that parameter).
85         */
86
87         if (rgelt == 0) return COM.E_INVALIDARG;
88         if (pceltFetched == 0 && celt != 1) return COM.E_INVALIDARG;
89
90         FORMATETC[] nextItems = getNextItems(celt);
91         if (nextItems != null) {
92                 for (int i = 0; i < nextItems.length; i++) {
93                         COM.MoveMemory(rgelt + i*FORMATETC.sizeof, nextItems[i], FORMATETC.sizeof);
94                 }
95
96                 if (pceltFetched != 0)
97                         OS.MoveMemory(pceltFetched, new int[] {nextItems.length}, 4);
98
99                 if (nextItems.length == celt) return COM.S_OK;
100
101         } else {
102                 if (pceltFetched != 0)
103                         OS.MoveMemory(pceltFetched, new int[] {0}, 4);
104                 COM.MoveMemory(rgelt, new FORMATETC(), FORMATETC.sizeof);
105
106         }
107         return COM.S_FALSE;
108 }
109 private int QueryInterface(long riid, long ppvObject) {
110
111         if (riid == 0 || ppvObject == 0) return COM.E_NOINTERFACE;
112
113         GUID guid = new GUID();
114         COM.MoveMemory(guid, riid, GUID.sizeof);
115
116         if (COM.IsEqualGUID(guid, COM.IIDIUnknown) || COM.IsEqualGUID(guid, COM.IIDIEnumFORMATETC)) {
117                 OS.MoveMemory(ppvObject, new long[] {iEnumFORMATETC.getAddress()}, C.PTR_SIZEOF);
118                 AddRef();
119                 return COM.S_OK;
120         }
121         OS.MoveMemory(ppvObject, new long[] {0}, C.PTR_SIZEOF);
122         return COM.E_NOINTERFACE;
123 }
124 int Release() {
125         refCount--;
126
127         if (refCount == 0) {
128                 disposeCOMInterfaces();
129                 if (COM.FreeUnusedLibraries) {
130                         COM.CoFreeUnusedLibraries();
131                 }
132         }
133
134         return refCount;
135 }
136 private int Reset() {
137         //Resets the enumeration sequence to the beginning.
138         index = 0;
139         return COM.S_OK;
140 }
141 void setFormats(FORMATETC[] newFormats) {
142         formats = newFormats;
143         index = 0;
144 }
145 private int Skip(int celt) {
146         //Skips over the next specified number of elements in the enumeration sequence.
147         if (celt < 1 ) return COM.E_INVALIDARG;
148
149         index += celt;
150         if (index > (formats.length - 1)){
151                 index = formats.length - 1;
152                 return COM.S_FALSE;
153         }
154         return COM.S_OK;
155 }
156 }