/Visual Studio 2008/CppAutomateExcel/CppAutomateExcel.cpp

# · C++ · 161 lines · 58 code · 17 blank · 86 comment · 9 complexity · 5c20916072d407dea888ab9c39ace98d MD5 · raw file

  1. /****************************** Module Header ******************************\
  2. * Module Name: CppAutomateExcel.cpp
  3. * Project: CppAutomateExcel
  4. * Copyright (c) Microsoft Corporation.
  5. *
  6. * The CppAutomateExcel example demonstrates how to write VC++ codes to create
  7. * a Microsoft Excel instance, create a workbook, fill data into a specific
  8. * range, save the workbook, close the Microsoft Excel application and then
  9. * clean up unmanaged COM resources.
  10. *
  11. * There are three basic ways you can write VC++ automation codes:
  12. *
  13. * 1. Automating Excel using the #import directive and smart pointers
  14. * (Solution1.h/cpp)
  15. * 2. Automating Excel using C++ and the COM APIs (Solution2.h/cpp)
  16. * 3. Automating Excel using MFC (This is not covered in this sample)
  17. *
  18. * This source is subject to the Microsoft Public License.
  19. * See http://www.microsoft.com/opensource/licenses.mspx#Ms-PL.
  20. * All other rights reserved.
  21. *
  22. * THIS CODE AND INFORMATION IS PROVIDED "AS IS" WITHOUT WARRANTY OF ANY KIND,
  23. * EITHER EXPRESSED OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE IMPLIED
  24. * WARRANTIES OF MERCHANTABILITY AND/OR FITNESS FOR A PARTICULAR PURPOSE.
  25. \***************************************************************************/
  26. #pragma region Includes
  27. #include <stdio.h>
  28. #include <windows.h>
  29. #include "Solution1.h" // The example of using the #import directive
  30. // and smart pointers to automate Excel
  31. #include "Solution2.h" // The example of using the raw COM API to
  32. // automate Excel
  33. #pragma endregion
  34. int wmain(int argc, wchar_t* argv[])
  35. {
  36. HANDLE hThread;
  37. // Demonstrate automating Excel using the #import directive and smart
  38. // pointers in a separate thread.
  39. hThread = CreateThread(NULL, 0, AutomateExcelByImport, NULL, 0, NULL);
  40. WaitForSingleObject(hThread, INFINITE);
  41. CloseHandle(hThread);
  42. _putws(L"");
  43. // Demonstrate automating Word using C++ and the COM APIs in a separate
  44. // thread.
  45. hThread = CreateThread(NULL, 0, AutomateExcelByCOMAPI, NULL, 0, NULL);
  46. WaitForSingleObject(hThread, INFINITE);
  47. CloseHandle(hThread);
  48. return 0;
  49. }
  50. //
  51. // FUNCTION: GetModuleDirectory(LPWSTR, DWORD);
  52. //
  53. // PURPOSE: This is a helper function in this sample. It retrieves the
  54. // fully-qualified path for the directory that contains the executable
  55. // file of the current process. For example, "D:\Samples\".
  56. //
  57. // PARAMETERS:
  58. // * pszDir - A pointer to a buffer that receives the fully-qualified
  59. // path for the directory taht contains the executable file of the
  60. // current process. If the length of the path is less than the size that
  61. // the nSize parameter specifies, the function succeeds and the path is
  62. // returned as a null-terminated string.
  63. // * nSize - The size of the lpFilename buffer, in characters.
  64. //
  65. // RETURN VALUE: If the function succeeds, the return value is the length
  66. // of the string that is copied to the buffer, in characters, not
  67. // including the terminating null character. If the buffer is too small
  68. // to hold the directory name, the function returns 0 and sets the last
  69. // error to ERROR_INSUFFICIENT_BUFFER. If the function fails, the return
  70. // value is 0 (zero). To get extended error information, call
  71. // GetLastError.
  72. //
  73. DWORD GetModuleDirectory(LPWSTR pszDir, DWORD nSize)
  74. {
  75. // Retrieve the path of the executable file of the current process.
  76. nSize = GetModuleFileName(NULL, pszDir, nSize);
  77. if (!nSize || GetLastError() == ERROR_INSUFFICIENT_BUFFER)
  78. {
  79. *pszDir = L'\0'; // Ensure it's NULL terminated
  80. return 0;
  81. }
  82. // Run through looking for the last slash in the file path.
  83. // When we find it, NULL it to truncate the following filename part.
  84. for (int i = nSize - 1; i >= 0; i--)
  85. {
  86. if (pszDir[i] == L'\\' || pszDir[i] == L'/')
  87. {
  88. pszDir[i + 1] = L'\0';
  89. nSize = i + 1;
  90. break;
  91. }
  92. }
  93. return nSize;
  94. }
  95. //
  96. // FUNCTION: SafeArrayPutName(SAFEARRAY*, long, PCWSTR, PCWSTR);
  97. //
  98. // PURPOSE: This is a helper function in the sample. It puts a user name
  99. // (first name, last name) into a 2D safe array. The array is in this
  100. // form:
  101. //
  102. // John Smith
  103. // Tom Brown
  104. // Sue Thomas
  105. //
  106. // Value in the first column is specified by pszFirstName. Value in the
  107. // second column is specified by pszLastName. SafeArrayPutName is used
  108. // add one entry (pszFirstName pszLastName) to the array on the row
  109. // indicated by the index parameter.
  110. //
  111. // PARAMETERS:
  112. // * psa - Pointer to an array descriptor created by SafeArrayCreate.
  113. // * index - The index of the name (first name, last name) in the array.
  114. // i.e. the first dimension of the 2D array.
  115. // * pszFirstName - The first name.
  116. // * pszLastName - The last name.
  117. //
  118. // RETURN VALUE: An HRESULT value indicating whether the function succeeds
  119. // or not.
  120. //
  121. HRESULT SafeArrayPutName(SAFEARRAY* psa, long index, PCWSTR pszFirstName,
  122. PCWSTR pszLastName)
  123. {
  124. HRESULT hr;
  125. // Set the first name.
  126. long indices1[] = { index, 1 };
  127. VARIANT vtFirstName;
  128. vtFirstName.vt = VT_BSTR;
  129. vtFirstName.bstrVal = SysAllocString(pszFirstName);
  130. // Copies the VARIANT into the SafeArray
  131. hr = SafeArrayPutElement(psa, indices1, (void*)&vtFirstName);
  132. VariantClear(&vtFirstName);
  133. if (SUCCEEDED(hr))
  134. {
  135. // Next, set the last name.
  136. long indices2[] = { index, 2 };
  137. VARIANT vtLastName;
  138. vtLastName.vt = VT_BSTR;
  139. vtLastName.bstrVal = SysAllocString(pszLastName);
  140. hr = SafeArrayPutElement(psa, indices2, (void*)&vtLastName);
  141. VariantClear(&vtLastName);
  142. }
  143. return hr;
  144. }