|
Hello,
i created a shell extension for my program but i cannot delete them? Whats going wrong there? Hope someone can help me. Thanks !
<br />
HKEY key;<br />
DWORD res;<br />
<br />
<br />
RegCreateKeyEx(HKEY_CLASSES_ROOT,L"CPACK\\shell\\Extract with CPACK\\command",0,NULL,REG_OPTION_NON_VOLATILE,0xF003F,NULL,&key,&res);<br />
wchar_t *data = L"C:\\cpack.exe e %1";<br />
RegSetValueEx(key,L"",0,REG_SZ,(LPBYTE)data,(DWORD)(lstrlen(data)+1)*sizeof(TCHAR));<br />
RegCloseKey(key);<br />
<br />
RegCreateKeyEx(HKEY_CLASSES_ROOT,L".cpack",0,NULL,REG_OPTION_NON_VOLATILE,0xF003F,NULL,&key,&res);<br />
data = L"CPACK";<br />
RegSetValueEx(key,L"",0,REG_SZ,(LPBYTE)data,(DWORD)(lstrlen(data)+1)*sizeof(TCHAR));<br />
RegCloseKey(key);<br />
<br />
<br />
RegDeleteKey(HKEY_CLASSES_ROOT,L"CPACK");<br />
bye,
gabbana
|
|
|
|
|
You cannot delete a registry key if subkeys exist.
If you were successful with creating the key and subkeys then you should be able to delete them as well. RegDeleteKey will fail with ERROR_ACCESS_DENIED if a subkey exists. You need to implement a recursive deletion[^].
Best Wishes,
-David Delaune
|
|
|
|
|
ahh okay now it works fine. Thanks.
|
|
|
|
|
gabbana wrote: Whats going wrong there?
What's going on is that you are not checking for errors. From MSDN:
If RegDeleteKey() fails, the return value is a nonzero error code defined in Winerror.h. You can use the FormatMessage() function with the FORMAT_MESSAGE_FROM_SYSTEM flag to get a generic description of the error.
"Love people and use things, not love things and use people." - Unknown
"To have a respect for ourselves guides our morals; to have deference for others governs our manners." - Laurence Sterne
|
|
|
|
|
|
A Visual C++ program of mine has been suffering from an ailment that it did not have under Borland C++ on my old PC: ignoring the keyboard, unprogrammed bell beeping when I press a keyboard key. I may have tracked it down, all or partly.
It has an array char KEYS[256] which is used as a boolean array to keep track of which keyboard keys are up and which are down. When I switch away from the program using alt-tab, this seems to happen: Alt down: is recorded. Tab down: the program loses the foreground. Alt up and tab up: not recorded by my program. So, when my program gets the foreground back, it thinks that the alt key is still down.
What I want to do is to trap any system message that says "This program is about to lose the foreground", so I can set to 0 the variable KEYS[VK_MENU] and any other KEYS elements that are so affected.
VK_MENU is the virtual key number of the ALT key.
|
|
|
|
|
You can probably use the WM_KILLFOCUS[^] message to detect when your window loses the keyboard focus.
You could also possibly use WM_ACTIVATE[^] to detect when your application has been activated or inactivated.
Best Wishes,
-David Delaune
|
|
|
|
|
Thanks. I found that:
case WM_KILLFOCUS: for(i=0;i<256;i++) KEYS[i]=0; goto DEF; seems to make dialog boxes misbehave.
case WM_ACTIVATE: for(i=0;i<256;i++) KEYS[i]=0; goto DEF; seems so far to work OK.
|
|
|
|
|
I have a Windows service in which I need to stop worker threads from my main thread. I am trying to use events to do this but I am stumped on why it is not working. Each class that runs a worker thread creates a stop event and a response event. When I need to stop the worker thread I call into the threads class and set the event. However, instead of the worker thread detecting that the event has been set and stopping gracefully, it just disappears. Why?
What is the best method for stopping a worker thread from the main thread?
Re-posted from the C# forum.
|
|
|
|
|
This should work quite well. I do it all the time. Something else is wrong.
Of course, your thread must periodically be 'waiting'on the event. If you require a polled approach because your thread is always working and does not return to check for the 'main event' too often, then you can use InterLockedIncrement and InterlockedCompareExchange on a control variable.
|
|
|
|
|
I appreciate any help. I have posted my code below:
<br />
void CFileChangeEvent::stopProcessThread()<br />
{<br />
TRACE( "> > > > CFileChangeEvent::stopProcessThread() entered - Event handles < %X, %X >, Event ID < %s >\n",<br />
m_hProcessStopEvent, m_hProcessRespEvent, getIdentifier() );<br />
<br />
::SetEvent( m_hProcessStopEvent );<br />
}<br />
<br />
UINT CFileChangeEvent::MultiFileProcessThread( LPVOID lpParam )<br />
{<br />
CFileChangeEvent* pEvent = (CFileChangeEvent*)lpParam;<br />
<br />
CFlightDataHandler* pFlightDataHandler = NULL;<br />
CLogWriter* pLogWriter = NULL;<br />
HANDLE hProcessStopEvent = pEvent->getProcessStopEvent();<br />
HANDLE hProcessRespEvent = pEvent->getProcessRespEvent();<br />
<br />
try<br />
{<br />
CDataMovement* pDataMovement = pEvent->getDataMovementParent();<br />
CLogWriter* pLogWriter = pDataMovement->getLogWriter();<br />
pFlightDataHandler = new CFlightDataHandler( pLogWriter );<br />
<br />
HRESULT hRes = ::CoInitialize(NULL);<br />
<br />
CString strInboxPath( pDataMovement->getInboxPath() );<br />
vector<CString> vecFifFiles = UtilK::findAllFiles( "*.fif", strInboxPath );<br />
<br />
int nLen = strInboxPath.GetLength() + 1;<br />
vector<CString>::iterator iterFile = vecFifFiles.begin();<br />
CString strFile = *iterFile;<br />
CString strLastPath = (*iterFile).Mid( 0, strFile.Find( "\\", nLen ) );<br />
<br />
int nFileCount = 0;<br />
bool bCancelled = false;<br />
for( ; iterFile != vecFifFiles.end() && ! bCancelled; iterFile++ )<br />
{<br />
CString strPath = UtilK::getFsPart( *iterFile, UtilK::FsPath );<br />
pFlightDataHandler->importRawFlightData( strPath );<br />
<br />
DWORD dwWaitStatus = ::WaitForSingleObject( hProcessStopEvent, 1 ); <br />
if ( dwWaitStatus == WAIT_FAILED )<br />
bCancelled = true;<br />
else if ( dwWaitStatus == WAIT_OBJECT_0 )<br />
bCancelled = true;<br />
}<br />
}<br />
catch( CMultiLevelException e )<br />
{<br />
CString strMsg;<br />
strMsg.Format( "importVARFile() failed - error description is %s", e.getUserMessage() );<br />
<br />
TRACE( "> > > > CFileChangeEvent::MultiFileProcessThread() exception < %s >\n", strMsg );<br />
if ( pLogWriter )<br />
pLogWriter->writeEntry( "CFileChangeEvent", CLogWriter::LE_OPERATION_FAILED, strMsg );<br />
}<br />
catch(...)<br />
{<br />
CString strMsg( "importVARFile() failed with unexpected exception" );<br />
TRACE( "> > > > CFileChangeEvent::MultiFileProcessThread() exception < %s >\n", strMsg );<br />
if ( pLogWriter )<br />
pLogWriter->writeEntry( "CVarFileEvent", CLogWriter::LE_OPERATION_FAILED, strMsg );<br />
}<br />
<br />
if ( pFlightDataHandler )<br />
delete pFlightDataHandler;<br />
<br />
CFileChangeEvent::setProcessCompleted();<br />
<br />
TRACE( "> > > > CFileChangeEvent::MultiFileProcessThread() exiting\n" );<br />
<br />
::CoUninitialize();<br />
<br />
if ( hProcessStopEvent )<br />
::CloseHandle( hProcessStopEvent );<br />
<br />
if ( hProcessRespEvent )<br />
{<br />
::SetEvent( hProcessRespEvent );<br />
::CloseHandle( hProcessRespEvent );<br />
}<br />
<br />
::AfxEndThread( 0 );<br />
<br />
return 1;<br />
}<br />
|
|
|
|
|
A couple things I see going on here.
1. You should verify pFlightDataHandler got created before using it.
2. I am assuming that the calls to getProcessStopEvent and getProcessRespEvent actually open a handle, otherwise, when you close them at the end of your thread, you have closed them off for the 'parent' as well.
3. It is not necessary to call AfxEndThread if you are going to exit the thread via its return. Just return the value 0 at end of thread function to accomplish same result.
4. I am assuming that pLogWriter's writeEntry will not throw exceptions, otherwise, you could just flag a problem in your exception handling code and then write to log after leaving the exception.
5. I would consider moving the CoInitialize ahead of your main try block, since the CoUninitialize is already outside of it.
6. If the pFlightDataHandler is a 'risky' object, consider putting its delete in a try/catch block.
7. Why not wait for 0 milliseconds, as per MSDN The function returns if the interval elapses, even if the object's state is nonsignaled. If dwMilliseconds is zero, the function tests the object's state and returns immediately. Then I think it might not even suspend your thread if the system is not busy.
Otherwise, I am not sure why you would have a problem, unless one of your obejcts is bad, as there are a couple other retrieval functions there where the pointers are not verified against NULL.
|
|
|
|
|
I've installed "Visual C++ 2008 feature pack", but the documents couldn't be updated.
The documents are already on www.msdn.com[^], any way to make it local?
|
|
|
|
|
Did you install MSDN on your system and it doesnt work?
|
|
|
|
|
The most up-to-date docs for the feature pack are only available online
unfortunately. Hopefully they'll make them available for download in
the near future.
Mark
Mark Salsbery
Microsoft MVP - Visual C++
|
|
|
|
|
I've got a problem that I've tried to figure out, but can't.
Here's my code:
CString str = "C:\\Wells\\Program\\DataBases\\Drivers\\R and S\\OurDrivers.ddb";
CStdioFile myFile;
if(!myFile.Open(str, CFile::modeRead | CFile::typeText))
{
TRACE(_T("Unable to open file\n"));
}
DWORD dw = GetLastError();// error_path_not_found (3)
Here's the error with the CString str
ERROR:
error C2440: 'initializing' : cannot convert from 'const char [58]' to 'ATL::CStringT<BaseType,StringTraits>'
now I realize that this is correct:
CString str = _T("C:\\Wells\\Program\\DataBases\\Drivers\\R and S\\OurDrivers.ddb");
Here's the problem. I have to send "str" to the FileSave" function with the data already in it! I can't convert it like above (hard-coded) because I don't know what cchoices the User used to build the above string(str). I just hard-coded it for testing purposes.
Isn't there someway to use a CString as the filename/path in CStudioFile?
Help Please!
A C++ programming language novice, but striving to learn
|
|
|
|
|
Larry Mills Sr wrote: ...the FileSave" function...
What does its signature look like?
Larry Mills Sr wrote: I just hard-coded it for testing purposes.
So str will actually be populated by the user?
Larry Mills Sr wrote: Isn't there someway to use a CString as the filename/path in CStudioFile?
Yes, and it normally does not require anything special.
"Love people and use things, not love things and use people." - Unknown
"To have a respect for ourselves guides our morals; to have deference for others governs our manners." - Laurence Sterne
|
|
|
|
|
You can use the ATL and MFC String Conversion Macros[^] to convert from wide string to LPCTSTR. Note that it is normally not recommended to use CStringA directly. In addition it is also not recommended to call the GetBuffer() member when there are better alternatives. However with those disclaimers out of the way lets continue to hammer a square peg into a round hole.
An ugly dirty method would be:
CString str = _T("C:\\Wells\\Program\\DataBases\\Drivers\\R and S\\OurDrivers.ddb");
USES_CONVERSION;
myFile.Open((LPCTSTR)CStringA(T2A(str)).GetBuffer(),0,0);
A better way of doing this would be:
CString str = _T("C:\\Wells\\Program\\DataBases\\Drivers\\R and S\\OurDrivers.ddb");
char szPath[MAX_PATH + 1] = {0};
USES_CONVERSION;
strcpy(szPath,T2A(str));
myFile.Open((LPCTSTR)szPath,0,0);
An even better way of doing this might be:
#ifdef _UNICODE
char szPath2[MAX_PATH + 1] = {0};
WideCharToMultiByte(CP_ACP, 0, str, -1, szPath2, MAX_PATH, NULL, NULL);
myFile.Open((LPCTSTR)szPath,0,0);
#endif
Best Wishes,
-David Delaune
|
|
|
|
|
Larry Mills Sr wrote: I have to send "str" to the FileSave" function with the data already in it! I can't convert it like above (hard-coded) because I don't know what cchoices the User used to build the above string(str).
That seems FileSave functions takes CString as parameter, then why do you worry, it is the user who has to ensure proper data in str expected by FileSave method.
Answer DavidCrow's query, don't think about conversion unless you know what you are doing.
|
|
|
|
|
1. I'm using GDI+, when I press F1 to see the help of Pen , it'll try to find Gdiplus::Pen , but find nothing. If I typed Pen manually, it'll be found.
Any way to let F1 find it automatically?
2.
Pen pen(Color::Red);
When compiling with warning level 4, it reports a warning as below:
warning C4245: 'argument' : conversion from '' to 'Gdiplus::ARGB', signed/unsigned mismatch
What a type mismatch!
How to write it correctly?
|
|
|
|
|
hi to all
i am a new bie
i want to know usage of GetDetaitsOf() , NameSpace() , ParseName()
there are JScript , VBScript , Visual Basic examples
on MSDN[^] but i need a C++ example can you show me one
thanks
|
|
|
|
|
reteset wrote: hi to all
i am a new bie
Then you should be studying C++ fundamentals not trying to reproduce code from Javascript and other languages.
led mike
|
|
|
|
|
you probably need to use the IShellFolder2 Interface.
|
|
|
|
|
reteset wrote: i am a new bie
hope the following gives a start, assuming you are not c/c++ newbie but the shell interface newbie,
C++ example for the example[^],
#include <shlobj.h>
#include <shlwapi.h>
#include <stdio.h>
#include <tchar.h>
#pragma comment (lib, "shlwapi.lib")
#define SAFE_RELEASE(_x) if (_x){_x->Release(); _x = NULL;}
int main()
{
LPMALLOC pMalloc = NULL;
IShellFolder *psfShell = NULL;
IShellFolder2 *psfWindowsFolder = NULL;
LPITEMIDLIST pidlWindowsFolder = NULL;
LPITEMIDLIST pidlFile = NULL;
TCHAR szDisplayName[MAX_PATH];
SHELLDETAILS shDetails;
HRESULT hr = E_FAIL;
hr = SHGetMalloc(&pMalloc);
if(FAILED(hr))
goto END;
hr = SHGetDesktopFolder(&psfShell);
if(FAILED(hr))
goto END;
hr = SHGetFolderLocation(NULL, CSIDL_WINDOWS, NULL, NULL,
&pidlWindowsFolder);
if(FAILED(hr))
goto END;
hr = psfShell->BindToObject(pidlWindowsFolder, NULL,
IID_IShellFolder2, (VOID **)&psfWindowsFolder);
if(FAILED(hr))
goto END;
hr = psfWindowsFolder->ParseDisplayName(NULL,
NULL,
_T("clock.avi"),
NULL,
&pidlFile,
NULL);
if(FAILED(hr))
goto END;
hr = psfWindowsFolder->GetDetailsOf(pidlFile, 2, &shDetails);
if(FAILED(hr))
goto END;
hr = StrRetToBuf(&shDetails.str, pidlFile, szDisplayName, sizeof(szDisplayName));
if(FAILED(hr))
goto END;
_tprintf(_T("Details at Column 2 : %s \n"), szDisplayName);
END:
if (pMalloc)
{
if (pidlFile) pMalloc->Free(pidlFile);
if (pidlWindowsFolder) pMalloc->Free(pidlWindowsFolder);
}
SAFE_RELEASE(psfWindowsFolder);
SAFE_RELEASE(psfShell);
SAFE_RELEASE(pMalloc);
return 0;
}
modified on Friday, May 2, 2008 1:31 AM
|
|
|
|
|
Just curious... do you actually use the goto statement that much in production code?
Best Wishes,
-David Delaune
|
|
|
|
|