2013-07-31 3 views
3

Так что я пробовал MSAPI 5.4 TTS with event example. Теперь я создаю приложение подсказки cmd, которое использует SetNotifyCallbackFunction, но функция, которую я передаю, не вызывается. Я не эксперт в C++, поэтому мне трудно решить эту проблему, может ли кто-нибудь указать мне в правильном направлении или хотя бы дать мне хороший пример SetNotifyCallbackFunction?C++ MSAPI 5: SetNotifyCallbackFunction не работает

Вот упрощенная версия моего кода:

typedef void __stdcall SPNOTIFYCALLBACK(WPARAM wParam, LPARAM lParam); 

void __stdcall outsideeventFunction(WPARAM, LPARAM); 

void __stdcall outsideeventFunction(WPARAM wParam, LPARAM lParam){ 
    std::cout << "Event called::wParam: " << wParam << " lParam: " << lParam << std::endl; 
    SPEVENT eventItem; 
    memset(&eventItem, 0, sizeof(SPEVENT)); 
    while (SUCCEEDED(pV->GetEvents(1, &eventItem, NULL))) 
    { 
     bool exitNa = false; 
     switch (eventItem.eEventId) 
     { 
     case SPEI_WORD_BOUNDARY: 
      SPVOICESTATUS eventStatus; 
      pV->GetStatus(&eventStatus, NULL); 
      ULONG start, end; 
      start = eventStatus.ulInputWordPos; 
      end = eventStatus.ulInputWordLen; 
      std::cout << "From event Test: " << start << ", " << end << std::endl; 
      std::cout << "From event Length: " << theString.length() - 1 << ", " << start + end << std::endl; 
      if (theString.length() - 1 <= start + end){ 
       std::cout << "From event Exit!" << std::endl; 
       exitNa = true; 
      } 
      break; 
     } 

     SpClearEvent(&eventItem); 
     if (exitNa){ 
      break; 
     } 
    } 
    return; 
} 

int _tmain(int argc, TCHAR* argv [], TCHAR* envp []) 
{ 
    pV = NULL; 
    std::string nativeString("Hello world, this is a test! For the purpose of a longer message, I'll add another sentence. And here comes the new sentence."); 
    SPNOTIFYCALLBACK *cb = &outsideeventFunction; 
    if (FAILED(::CoInitialize(NULL))) 
     return FALSE; 

    HRESULT hr = CoCreateInstance(CLSID_SpVoice, NULL, CLSCTX_ALL, IID_ISpVoice, (void **) &pV); 
    if (SUCCEEDED(hr)) 
    { 
     if (SUCCEEDED(pV->SetNotifyCallbackFunction(cb, 0, 0))){ 
      std::cout << "Success adding callback" << std::endl; 
     } 

     ULONGLONG ullMyEvents = SPFEI(SPEI_WORD_BOUNDARY); 
     pV->SetInterest(ullMyEvents, ullMyEvents); 
    } 

    theString = std::wstring(nativeString.begin(), nativeString.end()); 

    printf("Speak: %s\n", nativeString.c_str()); 
    hr = pV->Speak(theString.c_str(), SPF_ASYNC, NULL); 
    pV->WaitUntilDone(INFINITE); 

    std::system("pause"); 
    pV->Release(); 
    pV = NULL; 
    ::CoUninitialize(); 
    return TRUE; 
} 

Результат этого приложения является то, что синтезирование слова делается плавно, но outsideeventFunction никогда не была названа. Как вы видите, SetInterest правильно установлен. Как я могу это исправить?

+0

Какой голос вы используете? У некоторых (не Microsoft) голосов возникли проблемы с событиями .... –

ответ

3

Две проблемы:

1) SAPI не позволяют переназначить целевые события, так что линия

pV->SetNotifyWin32Event(); 

всегда будет терпеть неудачу.

2) Вам нужно перекачать сообщения, чтобы достать события. Таким образом, вместо вызова

pV->WaitUntilDone(INFINITE); 

вы должны получить ручку и насос сообщения, пока событие не будет установлено:

HANDLE hWait = pV->SpeakCompleteEvent(); 
WaitAndPumpMessagesWithTimeout(hWait, INFINITE); 

HRESULT WaitAndPumpMessagesWithTimeout(HANDLE hWaitHandle, DWORD dwMilliseconds) 
{ 
    HRESULT hr = S_OK; 
    BOOL fContinue = TRUE; 

    while (fContinue) 
    { 
     DWORD dwWaitId = ::MsgWaitForMultipleObjectsEx(1, &hWaitHandle, dwMilliseconds, QS_ALLINPUT, MWMO_INPUTAVAILABLE); 
     switch (dwWaitId) 
     { 
     case WAIT_OBJECT_0: 
      { 
       fContinue = FALSE; 
      } 
      break; 

     case WAIT_OBJECT_0 + 1: 
      { 
       MSG Msg; 
       while (::PeekMessage(&Msg, NULL, 0, 0, PM_REMOVE)) 
       { 
        ::TranslateMessage(&Msg); 
        ::DispatchMessage(&Msg); 
       } 
      } 
      break; 

     case WAIT_TIMEOUT: 
      { 
       hr = S_FALSE; 
       fContinue = FALSE; 
      } 
      break; 

     default:// Unexpected error 
      { 
       fContinue = FALSE; 
       hr = E_FAIL; 
      } 
      break; 
     } 
    } 
    return hr; 
} 
+0

О, я просто положил эту строку для экспериментов для этого случая. Даже без этого мероприятие все еще не работает. Есть еще предложения? – Marl

+1

Получил шанс просмотреть и добавил реальное исправление (вам нужно перекачать сообщения, чтобы получить доставленные события). –