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.
5 #include "base/memory/ref_counted.h"
6 #include "base/memory/scoped_handle.h"
7 #include "base/memory/scoped_ptr.h"
8 #include "base/strings/utf_string_conversions.h"
9 #include "base/win/scoped_hglobal.h"
10 #include "testing/gtest/include/gtest/gtest.h"
11 #include "ui/base/clipboard/clipboard.h"
12 #include "ui/base/dragdrop/os_exchange_data.h"
13 #include "ui/base/dragdrop/os_exchange_data_provider_win.h"
18 // Test getting using the IDataObject COM API
19 TEST(OSExchangeDataWinTest, StringDataAccessViaCOM) {
21 std::wstring input = L"O hai googlz.";
22 data.SetString(input);
23 base::win::ScopedComPtr<IDataObject> com_data(
24 OSExchangeDataProviderWin::GetIDataObject(data));
26 FORMATETC format_etc =
27 { CF_UNICODETEXT, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL };
28 EXPECT_EQ(S_OK, com_data->QueryGetData(&format_etc));
31 EXPECT_EQ(S_OK, com_data->GetData(&format_etc, &medium));
33 base::win::ScopedHGlobal<wchar_t>(medium.hGlobal).get();
34 EXPECT_EQ(input, output);
35 ReleaseStgMedium(&medium);
38 // Test setting using the IDataObject COM API
39 TEST(OSExchangeDataWinTest, StringDataWritingViaCOM) {
41 std::wstring input = L"http://www.google.com/";
43 base::win::ScopedComPtr<IDataObject> com_data(
44 OSExchangeDataProviderWin::GetIDataObject(data));
46 // Store data in the object using the COM SetData API.
47 CLIPFORMAT cfstr_ineturl = RegisterClipboardFormat(CFSTR_INETURL);
48 FORMATETC format_etc =
49 { cfstr_ineturl, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL };
51 medium.tymed = TYMED_HGLOBAL;
52 HGLOBAL glob = GlobalAlloc(GPTR, sizeof(wchar_t) * (input.size() + 1));
53 size_t stringsz = input.size();
54 SIZE_T sz = GlobalSize(glob);
55 base::win::ScopedHGlobal<wchar_t> global_lock(glob);
56 wchar_t* buffer_handle = global_lock.get();
57 wcscpy_s(buffer_handle, input.size() + 1, input.c_str());
58 medium.hGlobal = glob;
59 medium.pUnkForRelease = NULL;
60 EXPECT_EQ(S_OK, com_data->SetData(&format_etc, &medium, TRUE));
62 // Construct a new object with the old object so that we can use our access
64 OSExchangeData data2(data.provider().Clone());
65 EXPECT_TRUE(data2.HasURL(OSExchangeData::CONVERT_FILENAMES));
68 EXPECT_TRUE(data2.GetURLAndTitle(
69 OSExchangeData::CONVERT_FILENAMES, &url_from_data, &title));
70 GURL reference_url(input);
71 EXPECT_EQ(reference_url.spec(), url_from_data.spec());
74 // Verifies SetData invoked twice with the same data clobbers existing data.
75 TEST(OSExchangeDataWinTest, RemoveData) {
77 std::wstring input = L"http://www.google.com/";
78 std::wstring input2 = L"http://www.google2.com/";
80 base::win::ScopedComPtr<IDataObject> com_data(
81 OSExchangeDataProviderWin::GetIDataObject(data));
83 // Store data in the object using the COM SetData API.
84 CLIPFORMAT cfstr_ineturl = RegisterClipboardFormat(CFSTR_INETURL);
85 FORMATETC format_etc =
86 { cfstr_ineturl, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL };
88 medium.tymed = TYMED_HGLOBAL;
90 HGLOBAL glob = GlobalAlloc(GPTR, sizeof(wchar_t) * (input.size() + 1));
91 size_t stringsz = input.size();
92 SIZE_T sz = GlobalSize(glob);
93 base::win::ScopedHGlobal<wchar_t> global_lock(glob);
94 wchar_t* buffer_handle = global_lock.get();
95 wcscpy_s(buffer_handle, input.size() + 1, input.c_str());
96 medium.hGlobal = glob;
97 medium.pUnkForRelease = NULL;
98 EXPECT_EQ(S_OK, com_data->SetData(&format_etc, &medium, TRUE));
100 // This should clobber the existing data.
102 HGLOBAL glob = GlobalAlloc(GPTR, sizeof(wchar_t) * (input2.size() + 1));
103 size_t stringsz = input2.size();
104 SIZE_T sz = GlobalSize(glob);
105 base::win::ScopedHGlobal<wchar_t> global_lock(glob);
106 wchar_t* buffer_handle = global_lock.get();
107 wcscpy_s(buffer_handle, input2.size() + 1, input2.c_str());
108 medium.hGlobal = glob;
109 medium.pUnkForRelease = NULL;
110 EXPECT_EQ(S_OK, com_data->SetData(&format_etc, &medium, TRUE));
112 EXPECT_EQ(1u, static_cast<DataObjectImpl*>(com_data.get())->size());
114 // Construct a new object with the old object so that we can use our access
116 OSExchangeData data2(data.provider().Clone());
117 EXPECT_TRUE(data2.HasURL(OSExchangeData::CONVERT_FILENAMES));
120 EXPECT_TRUE(data2.GetURLAndTitle(
121 OSExchangeData::CONVERT_FILENAMES, &url_from_data, &title));
122 EXPECT_EQ(GURL(input2).spec(), url_from_data.spec());
125 TEST(OSExchangeDataWinTest, URLDataAccessViaCOM) {
127 GURL url("http://www.google.com/");
128 data.SetURL(url, L"");
129 base::win::ScopedComPtr<IDataObject> com_data(
130 OSExchangeDataProviderWin::GetIDataObject(data));
132 CLIPFORMAT cfstr_ineturl = RegisterClipboardFormat(CFSTR_INETURL);
133 FORMATETC format_etc =
134 { cfstr_ineturl, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL };
135 EXPECT_EQ(S_OK, com_data->QueryGetData(&format_etc));
138 EXPECT_EQ(S_OK, com_data->GetData(&format_etc, &medium));
139 std::wstring output =
140 base::win::ScopedHGlobal<wchar_t>(medium.hGlobal).get();
141 EXPECT_EQ(url.spec(), base::WideToUTF8(output));
142 ReleaseStgMedium(&medium);
145 TEST(OSExchangeDataWinTest, MultipleFormatsViaCOM) {
147 std::string url_spec = "http://www.google.com/";
149 std::wstring text = L"O hai googlz.";
150 data.SetURL(url, L"Google");
151 data.SetString(text);
153 base::win::ScopedComPtr<IDataObject> com_data(
154 OSExchangeDataProviderWin::GetIDataObject(data));
156 CLIPFORMAT cfstr_ineturl = RegisterClipboardFormat(CFSTR_INETURL);
157 FORMATETC url_format_etc =
158 { cfstr_ineturl, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL };
159 EXPECT_EQ(S_OK, com_data->QueryGetData(&url_format_etc));
160 FORMATETC text_format_etc =
161 { CF_UNICODETEXT, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL };
162 EXPECT_EQ(S_OK, com_data->QueryGetData(&text_format_etc));
165 EXPECT_EQ(S_OK, com_data->GetData(&url_format_etc, &medium));
166 std::wstring output_url =
167 base::win::ScopedHGlobal<wchar_t>(medium.hGlobal).get();
168 EXPECT_EQ(url.spec(), base::WideToUTF8(output_url));
169 ReleaseStgMedium(&medium);
171 // The text is supposed to be the raw text of the URL, _NOT_ the value of
172 // |text|! This is because the URL is added first and thus takes precedence!
173 EXPECT_EQ(S_OK, com_data->GetData(&text_format_etc, &medium));
174 std::wstring output_text =
175 base::win::ScopedHGlobal<wchar_t>(medium.hGlobal).get();
176 EXPECT_EQ(url_spec, base::WideToUTF8(output_text));
177 ReleaseStgMedium(&medium);
180 TEST(OSExchangeDataWinTest, EnumerationViaCOM) {
182 data.SetURL(GURL("http://www.google.com/"), L"");
183 data.SetString(L"O hai googlz.");
185 CLIPFORMAT cfstr_file_group_descriptor =
186 RegisterClipboardFormat(CFSTR_FILEDESCRIPTOR);
187 CLIPFORMAT text_x_moz_url = RegisterClipboardFormat(L"text/x-moz-url");
189 base::win::ScopedComPtr<IDataObject> com_data(
190 OSExchangeDataProviderWin::GetIDataObject(data));
191 base::win::ScopedComPtr<IEnumFORMATETC> enumerator;
192 EXPECT_EQ(S_OK, com_data.get()->EnumFormatEtc(DATADIR_GET,
193 enumerator.Receive()));
195 // Test that we can get one item.
197 // Explictly don't reset the first time, to verify the creation state is
200 FORMATETC elements_array[1];
201 EXPECT_EQ(S_OK, enumerator->Next(1,
202 reinterpret_cast<FORMATETC*>(&elements_array), &retrieved));
203 EXPECT_EQ(1, retrieved);
204 EXPECT_EQ(text_x_moz_url, elements_array[0].cfFormat);
207 // Test that we can get one item with a NULL retrieved value.
209 EXPECT_EQ(S_OK, enumerator->Reset());
210 FORMATETC elements_array[1];
211 EXPECT_EQ(S_OK, enumerator->Next(1,
212 reinterpret_cast<FORMATETC*>(&elements_array), NULL));
213 EXPECT_EQ(text_x_moz_url, elements_array[0].cfFormat);
216 // Test that we can get two items.
218 EXPECT_EQ(S_OK, enumerator->Reset());
220 FORMATETC elements_array[2];
221 EXPECT_EQ(S_OK, enumerator->Next(2,
222 reinterpret_cast<FORMATETC*>(&elements_array), &retrieved));
223 EXPECT_EQ(2, retrieved);
224 EXPECT_EQ(text_x_moz_url, elements_array[0].cfFormat);
225 EXPECT_EQ(cfstr_file_group_descriptor, elements_array[1].cfFormat);
228 // Test that we can skip the first item.
230 EXPECT_EQ(S_OK, enumerator->Reset());
231 EXPECT_EQ(S_OK, enumerator->Skip(1));
233 FORMATETC elements_array[1];
234 EXPECT_EQ(S_OK, enumerator->Next(1,
235 reinterpret_cast<FORMATETC*>(&elements_array), &retrieved));
236 EXPECT_EQ(1, retrieved);
237 EXPECT_EQ(cfstr_file_group_descriptor, elements_array[0].cfFormat);
240 // Test that we can skip the first item, and create a clone that matches in
241 // this state, and modify the original without affecting the clone.
243 EXPECT_EQ(S_OK, enumerator->Reset());
244 EXPECT_EQ(S_OK, enumerator->Skip(1));
245 base::win::ScopedComPtr<IEnumFORMATETC> cloned_enumerator;
246 EXPECT_EQ(S_OK, enumerator.get()->Clone(cloned_enumerator.Receive()));
247 EXPECT_EQ(S_OK, enumerator.get()->Reset());
251 FORMATETC elements_array[1];
252 EXPECT_EQ(S_OK, cloned_enumerator->Next(1,
253 reinterpret_cast<FORMATETC*>(&elements_array), &retrieved));
254 EXPECT_EQ(1, retrieved);
255 EXPECT_EQ(cfstr_file_group_descriptor, elements_array[0].cfFormat);
260 FORMATETC elements_array[1];
261 EXPECT_EQ(S_OK, enumerator->Next(1,
262 reinterpret_cast<FORMATETC*>(&elements_array), &retrieved));
263 EXPECT_EQ(1, retrieved);
264 EXPECT_EQ(text_x_moz_url, elements_array[0].cfFormat);
269 TEST(OSExchangeDataWinTest, TestURLExchangeFormatsViaCOM) {
271 std::string url_spec = "http://www.google.com/";
273 std::wstring url_title = L"www.google.com";
274 data.SetURL(url, url_title);
276 // File contents access via COM
277 base::win::ScopedComPtr<IDataObject> com_data(
278 OSExchangeDataProviderWin::GetIDataObject(data));
280 CLIPFORMAT cfstr_file_contents =
281 RegisterClipboardFormat(CFSTR_FILECONTENTS);
282 FORMATETC format_etc =
283 { cfstr_file_contents, NULL, DVASPECT_CONTENT, 0, TYMED_HGLOBAL };
284 EXPECT_EQ(S_OK, com_data->QueryGetData(&format_etc));
287 EXPECT_EQ(S_OK, com_data->GetData(&format_etc, &medium));
288 base::win::ScopedHGlobal<char> glob(medium.hGlobal);
289 std::string output(glob.get(), glob.Size());
290 std::string file_contents = "[InternetShortcut]\r\nURL=";
291 file_contents += url_spec;
292 file_contents += "\r\n";
293 EXPECT_EQ(file_contents, output);
294 ReleaseStgMedium(&medium);
298 TEST(OSExchangeDataWinTest, FileContents) {
300 std::string file_contents("data\0with\0nulls", 15);
301 data.SetFileContents(base::FilePath(L"filename.txt"), file_contents);
303 OSExchangeData copy(data.provider().Clone());
304 base::FilePath filename;
305 std::string read_contents;
306 EXPECT_TRUE(copy.GetFileContents(&filename, &read_contents));
307 EXPECT_EQ(L"filename.txt", filename.value());
308 EXPECT_EQ(file_contents, read_contents);
311 TEST(OSExchangeDataWinTest, CFHtml) {
313 GURL url("http://www.google.com/");
316 L"<b>bold.</b> <i><b>This is bold italic.</b></i>\n"
317 L"</BODY>\n</HTML>");
318 data.SetHtml(html, url);
320 // Check the CF_HTML too.
321 std::string expected_cf_html(
322 "Version:0.9\r\nStartHTML:0000000139\r\nEndHTML:0000000288\r\n"
323 "StartFragment:0000000175\r\nEndFragment:0000000252\r\n"
324 "SourceURL:http://www.google.com/\r\n<html>\r\n<body>\r\n"
325 "<!--StartFragment-->");
326 expected_cf_html += base::WideToUTF8(html);
327 expected_cf_html.append("<!--EndFragment-->\r\n</body>\r\n</html>");
329 FORMATETC format = Clipboard::GetHtmlFormatType().ToFormatEtc();
331 IDataObject* data_object = OSExchangeDataProviderWin::GetIDataObject(data);
332 EXPECT_EQ(S_OK, data_object->GetData(&format, &medium));
333 base::win::ScopedHGlobal<char> glob(medium.hGlobal);
334 std::string output(glob.get(), glob.Size());
335 EXPECT_EQ(expected_cf_html, output);
336 ReleaseStgMedium(&medium);
339 TEST(OSExchangeDataWinTest, SetURLWithMaxPath) {
341 std::wstring long_title(L'a', MAX_PATH + 1);
342 data.SetURL(GURL("http://google.com"), long_title);
345 TEST(OSExchangeDataWinTest, ProvideURLForPlainTextURL) {
347 data.SetString(L"http://google.com");
349 OSExchangeData data2(data.provider().Clone());
350 ASSERT_TRUE(data2.HasURL(OSExchangeData::CONVERT_FILENAMES));
353 EXPECT_TRUE(data2.GetURLAndTitle(
354 OSExchangeData::CONVERT_FILENAMES, &read_url, &title));
355 EXPECT_EQ(GURL("http://google.com"), read_url);