• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include "base/memory/ref_counted.h"
6 #include "base/memory/scoped_ptr.h"
7 #include "base/strings/utf_string_conversions.h"
8 #include "base/win/scoped_hglobal.h"
9 #include "testing/gtest/include/gtest/gtest.h"
10 #include "ui/base/clipboard/clipboard.h"
11 #include "ui/base/dragdrop/os_exchange_data.h"
12 #include "ui/base/dragdrop/os_exchange_data_provider_win.h"
13 #include "url/gurl.h"
14 
15 namespace ui {
16 
17 // Test getting using the IDataObject COM API
TEST(OSExchangeDataWinTest,StringDataAccessViaCOM)18 TEST(OSExchangeDataWinTest, StringDataAccessViaCOM) {
19   OSExchangeData data;
20   std::wstring input = L"O hai googlz.";
21   data.SetString(input);
22   base::win::ScopedComPtr<IDataObject> com_data(
23       OSExchangeDataProviderWin::GetIDataObject(data));
24 
25   FORMATETC format_etc =
26       { CF_UNICODETEXT, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL };
27   EXPECT_EQ(S_OK, com_data->QueryGetData(&format_etc));
28 
29   STGMEDIUM medium;
30   EXPECT_EQ(S_OK, com_data->GetData(&format_etc, &medium));
31   std::wstring output =
32       base::win::ScopedHGlobal<wchar_t*>(medium.hGlobal).get();
33   EXPECT_EQ(input, output);
34   ReleaseStgMedium(&medium);
35 }
36 
37 // Test setting using the IDataObject COM API
TEST(OSExchangeDataWinTest,StringDataWritingViaCOM)38 TEST(OSExchangeDataWinTest, StringDataWritingViaCOM) {
39   OSExchangeData data;
40   std::wstring input = L"http://www.google.com/";
41 
42   base::win::ScopedComPtr<IDataObject> com_data(
43       OSExchangeDataProviderWin::GetIDataObject(data));
44 
45   // Store data in the object using the COM SetData API.
46   CLIPFORMAT cfstr_ineturl = RegisterClipboardFormat(CFSTR_INETURL);
47   FORMATETC format_etc =
48       { cfstr_ineturl, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL };
49   STGMEDIUM medium;
50   medium.tymed = TYMED_HGLOBAL;
51   HGLOBAL glob = GlobalAlloc(GPTR, sizeof(wchar_t) * (input.size() + 1));
52   size_t stringsz = input.size();
53   SIZE_T sz = GlobalSize(glob);
54   base::win::ScopedHGlobal<wchar_t*> global_lock(glob);
55   wchar_t* buffer_handle = global_lock.get();
56   wcscpy_s(buffer_handle, input.size() + 1, input.c_str());
57   medium.hGlobal = glob;
58   medium.pUnkForRelease = NULL;
59   EXPECT_EQ(S_OK, com_data->SetData(&format_etc, &medium, TRUE));
60 
61   // Construct a new object with the old object so that we can use our access
62   // APIs.
63   OSExchangeData data2(data.provider().Clone());
64   EXPECT_TRUE(data2.HasURL(OSExchangeData::CONVERT_FILENAMES));
65   GURL url_from_data;
66   std::wstring title;
67   EXPECT_TRUE(data2.GetURLAndTitle(
68       OSExchangeData::CONVERT_FILENAMES, &url_from_data, &title));
69   GURL reference_url(input);
70   EXPECT_EQ(reference_url.spec(), url_from_data.spec());
71 }
72 
73 // Verifies SetData invoked twice with the same data clobbers existing data.
TEST(OSExchangeDataWinTest,RemoveData)74 TEST(OSExchangeDataWinTest, RemoveData) {
75   OSExchangeData data;
76   std::wstring input = L"http://www.google.com/";
77   std::wstring input2 = L"http://www.google2.com/";
78 
79   base::win::ScopedComPtr<IDataObject> com_data(
80       OSExchangeDataProviderWin::GetIDataObject(data));
81 
82   // Store data in the object using the COM SetData API.
83   CLIPFORMAT cfstr_ineturl = RegisterClipboardFormat(CFSTR_INETURL);
84   FORMATETC format_etc =
85       { cfstr_ineturl, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL };
86   STGMEDIUM medium;
87   medium.tymed = TYMED_HGLOBAL;
88   {
89     HGLOBAL glob = GlobalAlloc(GPTR, sizeof(wchar_t) * (input.size() + 1));
90     size_t stringsz = input.size();
91     SIZE_T sz = GlobalSize(glob);
92     base::win::ScopedHGlobal<wchar_t*> global_lock(glob);
93     wchar_t* buffer_handle = global_lock.get();
94     wcscpy_s(buffer_handle, input.size() + 1, input.c_str());
95     medium.hGlobal = glob;
96     medium.pUnkForRelease = NULL;
97     EXPECT_EQ(S_OK, com_data->SetData(&format_etc, &medium, TRUE));
98   }
99   // This should clobber the existing data.
100   {
101     HGLOBAL glob = GlobalAlloc(GPTR, sizeof(wchar_t) * (input2.size() + 1));
102     size_t stringsz = input2.size();
103     SIZE_T sz = GlobalSize(glob);
104     base::win::ScopedHGlobal<wchar_t*> global_lock(glob);
105     wchar_t* buffer_handle = global_lock.get();
106     wcscpy_s(buffer_handle, input2.size() + 1, input2.c_str());
107     medium.hGlobal = glob;
108     medium.pUnkForRelease = NULL;
109     EXPECT_EQ(S_OK, com_data->SetData(&format_etc, &medium, TRUE));
110   }
111   EXPECT_EQ(1u, static_cast<DataObjectImpl*>(com_data.get())->size());
112 
113   // Construct a new object with the old object so that we can use our access
114   // APIs.
115   OSExchangeData data2(data.provider().Clone());
116   EXPECT_TRUE(data2.HasURL(OSExchangeData::CONVERT_FILENAMES));
117   GURL url_from_data;
118   std::wstring title;
119   EXPECT_TRUE(data2.GetURLAndTitle(
120       OSExchangeData::CONVERT_FILENAMES, &url_from_data, &title));
121   EXPECT_EQ(GURL(input2).spec(), url_from_data.spec());
122 }
123 
TEST(OSExchangeDataWinTest,URLDataAccessViaCOM)124 TEST(OSExchangeDataWinTest, URLDataAccessViaCOM) {
125   OSExchangeData data;
126   GURL url("http://www.google.com/");
127   data.SetURL(url, L"");
128   base::win::ScopedComPtr<IDataObject> com_data(
129       OSExchangeDataProviderWin::GetIDataObject(data));
130 
131   CLIPFORMAT cfstr_ineturl = RegisterClipboardFormat(CFSTR_INETURL);
132   FORMATETC format_etc =
133       { cfstr_ineturl, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL };
134   EXPECT_EQ(S_OK, com_data->QueryGetData(&format_etc));
135 
136   STGMEDIUM medium;
137   EXPECT_EQ(S_OK, com_data->GetData(&format_etc, &medium));
138   std::wstring output =
139       base::win::ScopedHGlobal<wchar_t*>(medium.hGlobal).get();
140   EXPECT_EQ(url.spec(), base::WideToUTF8(output));
141   ReleaseStgMedium(&medium);
142 }
143 
TEST(OSExchangeDataWinTest,MultipleFormatsViaCOM)144 TEST(OSExchangeDataWinTest, MultipleFormatsViaCOM) {
145   OSExchangeData data;
146   std::string url_spec = "http://www.google.com/";
147   GURL url(url_spec);
148   std::wstring text = L"O hai googlz.";
149   data.SetURL(url, L"Google");
150   data.SetString(text);
151 
152   base::win::ScopedComPtr<IDataObject> com_data(
153       OSExchangeDataProviderWin::GetIDataObject(data));
154 
155   CLIPFORMAT cfstr_ineturl = RegisterClipboardFormat(CFSTR_INETURL);
156   FORMATETC url_format_etc =
157       { cfstr_ineturl, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL };
158   EXPECT_EQ(S_OK, com_data->QueryGetData(&url_format_etc));
159   FORMATETC text_format_etc =
160       { CF_UNICODETEXT, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL };
161   EXPECT_EQ(S_OK, com_data->QueryGetData(&text_format_etc));
162 
163   STGMEDIUM medium;
164   EXPECT_EQ(S_OK, com_data->GetData(&url_format_etc, &medium));
165   std::wstring output_url =
166       base::win::ScopedHGlobal<wchar_t*>(medium.hGlobal).get();
167   EXPECT_EQ(url.spec(), base::WideToUTF8(output_url));
168   ReleaseStgMedium(&medium);
169 
170   // The text is supposed to be the raw text of the URL, _NOT_ the value of
171   // |text|! This is because the URL is added first and thus takes precedence!
172   EXPECT_EQ(S_OK, com_data->GetData(&text_format_etc, &medium));
173   std::wstring output_text =
174       base::win::ScopedHGlobal<wchar_t*>(medium.hGlobal).get();
175   EXPECT_EQ(url_spec, base::WideToUTF8(output_text));
176   ReleaseStgMedium(&medium);
177 }
178 
TEST(OSExchangeDataWinTest,EnumerationViaCOM)179 TEST(OSExchangeDataWinTest, EnumerationViaCOM) {
180   OSExchangeData data;
181   data.SetURL(GURL("http://www.google.com/"), L"");
182   data.SetString(L"O hai googlz.");
183 
184   CLIPFORMAT cfstr_file_group_descriptor =
185       RegisterClipboardFormat(CFSTR_FILEDESCRIPTOR);
186   CLIPFORMAT text_x_moz_url = RegisterClipboardFormat(L"text/x-moz-url");
187 
188   base::win::ScopedComPtr<IDataObject> com_data(
189       OSExchangeDataProviderWin::GetIDataObject(data));
190   base::win::ScopedComPtr<IEnumFORMATETC> enumerator;
191   EXPECT_EQ(S_OK, com_data.get()->EnumFormatEtc(DATADIR_GET,
192                                                 enumerator.Receive()));
193 
194   // Test that we can get one item.
195   {
196     // Explictly don't reset the first time, to verify the creation state is
197     // OK.
198     ULONG retrieved = 0;
199     FORMATETC elements_array[1];
200     EXPECT_EQ(S_OK, enumerator->Next(1,
201         reinterpret_cast<FORMATETC*>(&elements_array), &retrieved));
202     EXPECT_EQ(1, retrieved);
203     EXPECT_EQ(text_x_moz_url, elements_array[0].cfFormat);
204   }
205 
206   // Test that we can get one item with a NULL retrieved value.
207   {
208     EXPECT_EQ(S_OK, enumerator->Reset());
209     FORMATETC elements_array[1];
210     EXPECT_EQ(S_OK, enumerator->Next(1,
211         reinterpret_cast<FORMATETC*>(&elements_array), NULL));
212     EXPECT_EQ(text_x_moz_url, elements_array[0].cfFormat);
213   }
214 
215   // Test that we can get two items.
216   {
217     EXPECT_EQ(S_OK, enumerator->Reset());
218     ULONG retrieved = 0;
219     FORMATETC elements_array[2];
220     EXPECT_EQ(S_OK, enumerator->Next(2,
221         reinterpret_cast<FORMATETC*>(&elements_array), &retrieved));
222     EXPECT_EQ(2, retrieved);
223     EXPECT_EQ(text_x_moz_url, elements_array[0].cfFormat);
224     EXPECT_EQ(cfstr_file_group_descriptor, elements_array[1].cfFormat);
225   }
226 
227   // Test that we can skip the first item.
228   {
229     EXPECT_EQ(S_OK, enumerator->Reset());
230     EXPECT_EQ(S_OK, enumerator->Skip(1));
231     ULONG retrieved = 0;
232     FORMATETC elements_array[1];
233     EXPECT_EQ(S_OK, enumerator->Next(1,
234         reinterpret_cast<FORMATETC*>(&elements_array), &retrieved));
235     EXPECT_EQ(1, retrieved);
236     EXPECT_EQ(cfstr_file_group_descriptor, elements_array[0].cfFormat);
237   }
238 
239   // Test that we can skip the first item, and create a clone that matches in
240   // this state, and modify the original without affecting the clone.
241   {
242     EXPECT_EQ(S_OK, enumerator->Reset());
243     EXPECT_EQ(S_OK, enumerator->Skip(1));
244     base::win::ScopedComPtr<IEnumFORMATETC> cloned_enumerator;
245     EXPECT_EQ(S_OK, enumerator.get()->Clone(cloned_enumerator.Receive()));
246     EXPECT_EQ(S_OK, enumerator.get()->Reset());
247 
248     {
249       ULONG retrieved = 0;
250       FORMATETC elements_array[1];
251       EXPECT_EQ(S_OK, cloned_enumerator->Next(1,
252           reinterpret_cast<FORMATETC*>(&elements_array), &retrieved));
253       EXPECT_EQ(1, retrieved);
254       EXPECT_EQ(cfstr_file_group_descriptor, elements_array[0].cfFormat);
255     }
256 
257     {
258       ULONG retrieved = 0;
259       FORMATETC elements_array[1];
260       EXPECT_EQ(S_OK, enumerator->Next(1,
261           reinterpret_cast<FORMATETC*>(&elements_array), &retrieved));
262       EXPECT_EQ(1, retrieved);
263       EXPECT_EQ(text_x_moz_url, elements_array[0].cfFormat);
264     }
265   }
266 }
267 
TEST(OSExchangeDataWinTest,TestURLExchangeFormatsViaCOM)268 TEST(OSExchangeDataWinTest, TestURLExchangeFormatsViaCOM) {
269   OSExchangeData data;
270   std::string url_spec = "http://www.google.com/";
271   GURL url(url_spec);
272   std::wstring url_title = L"www.google.com";
273   data.SetURL(url, url_title);
274 
275   // File contents access via COM
276   base::win::ScopedComPtr<IDataObject> com_data(
277       OSExchangeDataProviderWin::GetIDataObject(data));
278   {
279     CLIPFORMAT cfstr_file_contents =
280         RegisterClipboardFormat(CFSTR_FILECONTENTS);
281     FORMATETC format_etc =
282         { cfstr_file_contents, NULL, DVASPECT_CONTENT, 0, TYMED_HGLOBAL };
283     EXPECT_EQ(S_OK, com_data->QueryGetData(&format_etc));
284 
285     STGMEDIUM medium;
286     EXPECT_EQ(S_OK, com_data->GetData(&format_etc, &medium));
287     base::win::ScopedHGlobal<char*> glob(medium.hGlobal);
288     std::string output(glob.get(), glob.Size());
289     std::string file_contents = "[InternetShortcut]\r\nURL=";
290     file_contents += url_spec;
291     file_contents += "\r\n";
292     EXPECT_EQ(file_contents, output);
293     ReleaseStgMedium(&medium);
294   }
295 }
296 
TEST(OSExchangeDataWinTest,FileContents)297 TEST(OSExchangeDataWinTest, FileContents) {
298   OSExchangeData data;
299   std::string file_contents("data\0with\0nulls", 15);
300   data.SetFileContents(base::FilePath(L"filename.txt"), file_contents);
301 
302   OSExchangeData copy(data.provider().Clone());
303   base::FilePath filename;
304   std::string read_contents;
305   EXPECT_TRUE(copy.GetFileContents(&filename, &read_contents));
306   EXPECT_EQ(L"filename.txt", filename.value());
307   EXPECT_EQ(file_contents, read_contents);
308 }
309 
TEST(OSExchangeDataWinTest,CFHtml)310 TEST(OSExchangeDataWinTest, CFHtml) {
311   OSExchangeData data;
312   GURL url("http://www.google.com/");
313   std::wstring html(
314       L"<HTML>\n<BODY>\n"
315       L"<b>bold.</b> <i><b>This is bold italic.</b></i>\n"
316       L"</BODY>\n</HTML>");
317   data.SetHtml(html, url);
318 
319   // Check the CF_HTML too.
320   std::string expected_cf_html(
321       "Version:0.9\r\nStartHTML:0000000139\r\nEndHTML:0000000288\r\n"
322       "StartFragment:0000000175\r\nEndFragment:0000000252\r\n"
323       "SourceURL:http://www.google.com/\r\n<html>\r\n<body>\r\n"
324       "<!--StartFragment-->");
325   expected_cf_html += base::WideToUTF8(html);
326   expected_cf_html.append("<!--EndFragment-->\r\n</body>\r\n</html>");
327 
328   FORMATETC format = Clipboard::GetHtmlFormatType().ToFormatEtc();
329   STGMEDIUM medium;
330   IDataObject* data_object = OSExchangeDataProviderWin::GetIDataObject(data);
331   EXPECT_EQ(S_OK, data_object->GetData(&format, &medium));
332   base::win::ScopedHGlobal<char*> glob(medium.hGlobal);
333   std::string output(glob.get(), glob.Size());
334   EXPECT_EQ(expected_cf_html, output);
335   ReleaseStgMedium(&medium);
336 }
337 
TEST(OSExchangeDataWinTest,SetURLWithMaxPath)338 TEST(OSExchangeDataWinTest, SetURLWithMaxPath) {
339   OSExchangeData data;
340   std::wstring long_title(L'a', MAX_PATH + 1);
341   data.SetURL(GURL("http://google.com"), long_title);
342 }
343 
TEST(OSExchangeDataWinTest,ProvideURLForPlainTextURL)344 TEST(OSExchangeDataWinTest, ProvideURLForPlainTextURL) {
345   OSExchangeData data;
346   data.SetString(L"http://google.com");
347 
348   OSExchangeData data2(data.provider().Clone());
349   ASSERT_TRUE(data2.HasURL(OSExchangeData::CONVERT_FILENAMES));
350   GURL read_url;
351   std::wstring title;
352   EXPECT_TRUE(data2.GetURLAndTitle(
353       OSExchangeData::CONVERT_FILENAMES, &read_url, &title));
354   EXPECT_EQ(GURL("http://google.com"), read_url);
355 }
356 
357 }  // namespace ui
358