1 /* PROJECT: ReactOS sndrec32 2 * LICENSE: GPL - See COPYING in the top level directory 3 * FILE: base/applications/sndrec32/sndrec32.cpp 4 * PURPOSE: Sound recording 5 * PROGRAMMERS: Marco Pagliaricci (irc: rendar) 6 * Robert Naumann (gonzoMD) 7 */ 8 9 #include "stdafx.h" 10 11 #include <commctrl.h> 12 #include <commdlg.h> 13 #include <winnls.h> 14 15 #include "sndrec32.h" 16 #include "shellapi.h" 17 18 #ifndef _UNICODE 19 #define gprintf _snprintf 20 #else 21 #define gprintf _snwprintf 22 #endif 23 24 HINSTANCE hInst; 25 TCHAR szTitle[MAX_LOADSTRING]; 26 TCHAR szWindowClass[MAX_LOADSTRING]; 27 28 ATOM MyRegisterClass(HINSTANCE hInstance); 29 ATOM MyRegisterClass_wave(HINSTANCE hInstance); 30 BOOL InitInstance(HINSTANCE, int); 31 BOOL InitInstance_wave(HWND, HINSTANCE, int); 32 LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM); 33 LRESULT CALLBACK WndProc_wave(HWND, UINT, WPARAM, LPARAM); 34 35 BOOL win_first, wout_first; 36 37 HWND main_win; 38 HWND wave_win; 39 HWND slider; 40 HWND buttons[5]; 41 HBITMAP butbmps[5]; 42 HBITMAP butbmps_dis[5]; 43 WNDPROC buttons_std_proc; 44 45 BOOL butdisabled[5]; 46 BOOL stopped_flag; 47 BOOL isnew; 48 BOOL display_dur; 49 50 DWORD slider_pos; 51 WORD slider_min; 52 WORD slider_max; 53 54 DWORD samples_max; 55 56 OPENFILENAME ofn; 57 TCHAR file_path[MAX_PATH]; 58 TCHAR str_pos[MAX_LOADSTRING]; 59 TCHAR str_dur[MAX_LOADSTRING]; 60 TCHAR str_buf[MAX_LOADSTRING]; 61 TCHAR str_fmt[MAX_LOADSTRING]; 62 TCHAR str_chan[MAX_LOADSTRING]; 63 64 TCHAR str_mono[10]; 65 TCHAR str_stereo[10]; 66 67 BOOL path_set; 68 69 snd::audio_membuffer *AUD_BUF; 70 snd::audio_waveout *AUD_OUT; 71 snd::audio_wavein *AUD_IN; 72 73 BOOL s_recording; 74 75 NONCLIENTMETRICS s_info; 76 77 RECT text_rect; 78 RECT text2_rect; 79 RECT cli; 80 81 INT_PTR 82 CALLBACK 83 AboutDlgProc(HWND hWnd, 84 UINT msg, 85 WPARAM wp, 86 LPARAM lp) 87 { 88 switch (msg) 89 { 90 case WM_COMMAND: 91 switch (LOWORD(wp)) 92 { 93 case IDOK: 94 EndDialog(hWnd, 0); 95 return TRUE; 96 } 97 break; 98 case WM_CLOSE: 99 EndDialog(hWnd, 0); 100 return TRUE; 101 } 102 return FALSE; 103 } 104 105 int 106 APIENTRY 107 _tWinMain(HINSTANCE hInstance, 108 HINSTANCE hPrevInstance, 109 LPTSTR lpCmdLine, 110 int nCmdShow) 111 { 112 UNREFERENCED_PARAMETER(hPrevInstance); 113 UNREFERENCED_PARAMETER(lpCmdLine); 114 115 MSG msg; 116 HACCEL hAccelTable; 117 118 s_info.cbSize = sizeof( NONCLIENTMETRICS ); 119 120 InitCommonControls(); 121 122 switch (GetUserDefaultUILanguage()) 123 { 124 case MAKELANGID(LANG_HEBREW, SUBLANG_DEFAULT): 125 SetProcessDefaultLayout(LAYOUT_RTL); 126 break; 127 128 default: 129 break; 130 } 131 132 win_first = wout_first = FALSE; 133 134 text_rect.left = REFRESHA_X; 135 text_rect.top = REFRESHA_Y; 136 text_rect.right = REFRESHA_CX; 137 text_rect.bottom = REFRESHA_CY; 138 139 text2_rect.left = REFRESHB_X; 140 text2_rect.top = REFRESHB_Y; 141 text2_rect.right = REFRESHB_CX; 142 text2_rect.bottom = REFRESHB_CY; 143 144 /* Retrieving default system font, and others system informations */ 145 SystemParametersInfo(SPI_GETNONCLIENTMETRICS, 146 sizeof(NONCLIENTMETRICS), 147 &s_info, 148 0); 149 150 /* Set font size */ 151 s_info.lfMenuFont.lfHeight = 14; 152 153 /* Inits buttons bitmaps */ 154 155 butbmps[0] = LoadBitmap(hInstance, MAKEINTRESOURCE(IDB_BITMAP2_START)); 156 butbmps[1] = LoadBitmap(hInstance, MAKEINTRESOURCE(IDB_BITMAP2_END)); 157 butbmps[2] = LoadBitmap(hInstance, MAKEINTRESOURCE(IDB_BITMAP2_PLAY)); 158 butbmps[3] = LoadBitmap(hInstance, MAKEINTRESOURCE(IDB_BITMAP2_STOP)); 159 butbmps[4] = LoadBitmap(hInstance, MAKEINTRESOURCE(IDB_BITMAP2_REC)); 160 161 butbmps_dis[0] = LoadBitmap(hInstance, MAKEINTRESOURCE(IDB_BITMAP2_START_DIS)); 162 butbmps_dis[1] = LoadBitmap(hInstance, MAKEINTRESOURCE(IDB_BITMAP2_END_DIS)); 163 butbmps_dis[2] = LoadBitmap(hInstance, MAKEINTRESOURCE(IDB_BITMAP2_PLAY_DIS)); 164 butbmps_dis[3] = LoadBitmap(hInstance, MAKEINTRESOURCE(IDB_BITMAP2_STOP_DIS)); 165 butbmps_dis[4] = LoadBitmap(hInstance, MAKEINTRESOURCE(IDB_BITMAP2_REC_DIS)); 166 167 /* Inits audio devices and buffers */ 168 169 snd::audio_membuffer AUD_buffer(snd::A44100_16BIT_STEREO); 170 snd::audio_waveout AUD_waveout(snd::A44100_16BIT_STEREO, AUD_buffer); 171 snd::audio_wavein AUD_wavein(snd::A44100_16BIT_STEREO, AUD_buffer); 172 173 AUD_buffer.play_finished = l_play_finished; 174 AUD_buffer.audio_arrival = l_audio_arrival; 175 AUD_buffer.buffer_resized = l_buffer_resized; 176 177 AUD_buffer.alloc_seconds(INITIAL_BUFREC_SECONDS); 178 179 AUD_IN = &AUD_wavein; 180 AUD_OUT = &AUD_waveout; 181 AUD_BUF = &AUD_buffer; 182 183 /* Inits slider default parameters */ 184 185 slider_pos = 0; 186 slider_min = 0; 187 slider_max = SLIDER_W; 188 189 stopped_flag = FALSE; 190 path_set = FALSE; 191 isnew = TRUE; 192 display_dur = TRUE; 193 194 samples_max = AUD_buffer.total_samples(); 195 196 s_recording = false; 197 198 /* Inits strings */ 199 LoadString(hInstance, IDS_APP_TITLE, szTitle, MAX_LOADSTRING); 200 LoadString(hInstance, IDC_REACTOS_SNDREC32, szWindowClass, MAX_LOADSTRING); 201 LoadString(hInstance, IDS_STRPOS, str_pos, MAX_LOADSTRING); 202 LoadString(hInstance, IDS_STRDUR, str_dur, MAX_LOADSTRING); 203 LoadString(hInstance, IDS_STRBUF, str_buf, MAX_LOADSTRING); 204 LoadString(hInstance, IDS_STRFMT, str_fmt, MAX_LOADSTRING); 205 LoadString(hInstance, IDS_STRCHAN, str_chan, MAX_LOADSTRING); 206 LoadString(hInstance, IDS_STRMONO, str_mono, 10); 207 LoadString(hInstance, IDS_STRSTEREO, str_stereo, 10); 208 209 /* Registers sndrec32 window class */ 210 MyRegisterClass(hInstance); 211 MyRegisterClass_wave(hInstance); 212 213 if (!InitInstance(hInstance, nCmdShow)) 214 { 215 MessageBox(0, TEXT("CreateWindow() Error!"), TEXT("ERROR"), MB_ICONERROR); 216 return FALSE; 217 } 218 219 /* Loads key accelerators */ 220 hAccelTable = LoadAccelerators(hInstance, MAKEINTRESOURCE(IDC_REACTOS_SNDREC32)); 221 222 /* Starts main loop */ 223 while (GetMessage(&msg, NULL, 0, 0)) 224 { 225 if (!TranslateAccelerator(main_win, hAccelTable, &msg)) 226 { 227 TranslateMessage(&msg); 228 DispatchMessage(&msg); 229 } 230 } 231 232 if (wout_first) 233 { 234 AUD_waveout.close(); 235 } 236 237 if (win_first) 238 { 239 AUD_wavein.close(); 240 } 241 242 AUD_buffer.clear(); 243 244 return (int)msg.wParam; 245 } 246 247 ATOM 248 MyRegisterClass(HINSTANCE hInstance) 249 { 250 WNDCLASSEX wcex; 251 252 wcex.cbSize = sizeof(WNDCLASSEX); 253 254 wcex.style = CS_HREDRAW | CS_VREDRAW; 255 wcex.lpfnWndProc = WndProc; 256 wcex.cbClsExtra = 0; 257 wcex.cbWndExtra = 0; 258 wcex.hInstance = hInstance; 259 wcex.hIcon = LoadIcon(hInstance, MAKEINTRESOURCE(IDI_SNDREC32)); 260 wcex.hCursor = LoadCursor(NULL, IDC_ARROW); 261 wcex.hbrBackground = GetSysColorBrush(COLOR_BTNFACE); 262 wcex.lpszMenuName = MAKEINTRESOURCE(IDR_MENU1); 263 wcex.lpszClassName = szWindowClass; 264 wcex.hIconSm = LoadIcon(wcex.hInstance, MAKEINTRESOURCE(IDI_SNDREC32)); 265 266 return RegisterClassEx(&wcex); 267 } 268 269 BOOL 270 InitInstance(HINSTANCE hInstance, int nCmdShow) 271 { 272 HWND hWnd; 273 274 hInst = hInstance; 275 276 hWnd = CreateWindow(szWindowClass, 277 szTitle, 278 WS_OVERLAPPED | WS_CAPTION | WS_SYSMENU | WS_MINIMIZEBOX, 279 CW_USEDEFAULT, 280 CW_USEDEFAULT, 281 MAINWINDOW_W, 282 MAINWINDOW_H, 283 NULL, 284 NULL, 285 hInstance, 286 NULL); 287 if (!hWnd) 288 { 289 return FALSE; 290 } 291 292 ShowWindow(hWnd, nCmdShow); 293 UpdateWindow(hWnd); 294 295 main_win = hWnd; 296 297 return TRUE; 298 } 299 300 ATOM 301 MyRegisterClass_wave(HINSTANCE hInstance) 302 { 303 WNDCLASSEX wcex; 304 305 wcex.cbSize = sizeof(WNDCLASSEX); 306 307 wcex.style = CS_HREDRAW | CS_VREDRAW; 308 wcex.lpfnWndProc = WndProc_wave; 309 wcex.cbClsExtra = 0; 310 wcex.cbWndExtra = 0; 311 wcex.hInstance = hInstance; 312 wcex.hIcon = 0; 313 wcex.hCursor = LoadCursor(NULL, IDC_ARROW); 314 wcex.hbrBackground = (HBRUSH)GetStockObject(BLACK_BRUSH); 315 wcex.lpszMenuName = 0; 316 wcex.lpszClassName = TEXT("sndrec32_wave"); 317 wcex.hIconSm = 0; 318 319 return RegisterClassEx(&wcex); 320 } 321 322 BOOL 323 InitInstance_wave(HWND f, 324 HINSTANCE hInstance, 325 int nCmdShow) 326 { 327 HWND hWnd; 328 329 hInst = hInstance; 330 331 hWnd = CreateWindowEx(WS_EX_STATICEDGE, 332 TEXT("sndrec32_wave"), 333 TEXT(""), 334 WS_VISIBLE | WS_CHILD, 335 WAVEBAR_X, 336 WAVEBAR_Y, 337 WAVEBAR_CX, 338 WAVEBAR_CY, 339 f, 340 (HMENU)8, 341 hInstance, 342 0); 343 344 if (!hWnd ) 345 { 346 return FALSE; 347 } 348 349 ShowWindow(hWnd, nCmdShow); 350 UpdateWindow(hWnd); 351 352 wave_win = hWnd; 353 354 return TRUE; 355 } 356 357 LRESULT 358 CALLBACK 359 WndProc_wave(HWND hWnd, 360 UINT message, 361 WPARAM wParam, 362 LPARAM lParam) 363 { 364 PAINTSTRUCT ps; 365 HDC hdc; 366 HPEN pen; 367 HPEN oldpen; 368 369 unsigned int max_h = (cli.bottom / 2); 370 unsigned int samples; 371 unsigned int x, line_h; 372 373 switch (message) 374 { 375 case WM_CREATE: 376 GetClientRect(hWnd, &cli); 377 break; 378 379 case WM_PAINT: 380 /* Initialize hdc objects */ 381 hdc = BeginPaint(hWnd, &ps); 382 pen = (HPEN)CreatePen(PS_SOLID, 1, WAVEBAR_COLOR); 383 oldpen = (HPEN) SelectObject(hdc, (HBRUSH)pen); 384 if (AUD_OUT->current_status() == snd::WAVEOUT_PLAYING) 385 { 386 samples = AUD_OUT->tot_samples_buf(); 387 for (unsigned int i = 0; i < WAVEBAR_CX; ++i) 388 { 389 x = (i * samples) / WAVEBAR_CX; 390 line_h = (AUD_OUT->nsample(x) * max_h) / AUD_OUT->samplevalue_max(); 391 if (line_h) 392 { 393 MoveToEx(hdc, i, max_h, 0); 394 LineTo(hdc, i, max_h - (line_h * 2)); 395 LineTo(hdc, i, max_h + (line_h * 2)); 396 } 397 else 398 { 399 SetPixel(hdc, i, max_h, WAVEBAR_COLOR); 400 } 401 } 402 } 403 else if (AUD_IN->current_status() == snd::WAVEIN_RECORDING) 404 { 405 samples = AUD_IN->tot_samples_buf(); 406 for (unsigned int i = 0; i < WAVEBAR_CX; ++i) 407 { 408 x = (i * samples) / WAVEBAR_CX; 409 line_h = (AUD_IN->nsample(x) * max_h) / AUD_IN->samplevalue_max(); 410 if (line_h) 411 { 412 MoveToEx(hdc, i, max_h, 0); 413 LineTo(hdc, i, max_h - (line_h * 2)); 414 LineTo(hdc, i, max_h + (line_h * 2)); 415 } 416 else 417 { 418 SetPixel( hdc, i, max_h, WAVEBAR_COLOR ); 419 } 420 } 421 } 422 else 423 { 424 /* In standby mode draw a simple line */ 425 MoveToEx(hdc, 0, cli.bottom / 2, 0); 426 LineTo(hdc, WAVEBAR_CX, cli.bottom / 2); 427 } 428 429 SelectObject(hdc, oldpen); 430 DeleteObject( pen ); 431 EndPaint( hWnd, &ps ); 432 break; 433 434 case WM_USER: 435 break; 436 437 default: 438 return DefWindowProc(hWnd, message, wParam, lParam); 439 } 440 441 return 0; 442 } 443 444 LRESULT 445 CALLBACK 446 WndProc(HWND hWnd, 447 UINT message, 448 WPARAM wParam, 449 LPARAM lParam) 450 { 451 int wmId; 452 TCHAR str_tmp[MAX_LOADSTRING]; 453 PAINTSTRUCT ps; 454 HDC hdc; 455 HFONT font; 456 HFONT oldfont; 457 long long slid_samp = 0; 458 459 /* Checking for global pointers to buffer and io audio devices */ 460 if ((!AUD_IN) || (!AUD_OUT) || (!AUD_BUF)) 461 { 462 MessageBox(0, TEXT("Buffer Error"), 0, 0); 463 return 1; 464 } 465 466 switch (message) 467 { 468 case WM_CREATE: 469 /* Creating the wave bar */ 470 if (!InitInstance_wave(hWnd, hInst, SW_SHOWNORMAL)) 471 { 472 MessageBox(0, TEXT("CreateWindow() Error!"), TEXT("ERROR"), MB_ICONERROR); 473 return FALSE; 474 } 475 476 /* Creating ALL the buttons */ 477 for (int i = 0; i < 5; ++i) 478 { 479 buttons[i] = CreateWindow(TEXT("button"), 480 TEXT(""), 481 WS_CHILD | WS_VISIBLE | BS_BITMAP, 482 BUTTONS_CX + (i * (BUTTONS_W + ((i == 0) ? 0 : BUTTONS_SPACE))), 483 BUTTONS_CY, 484 BUTTONS_W, 485 BUTTONS_H, 486 hWnd, 487 (HMENU)i, 488 hInst, 489 0); 490 if (!buttons[i]) 491 { 492 MessageBox(0, 0, TEXT("CreateWindow() Error!"), 0); 493 return FALSE; 494 } 495 496 /* Realize the button bmp image */ 497 SendMessage(buttons[i], BM_SETIMAGE, (WPARAM)IMAGE_BITMAP, (LPARAM)butbmps[i]); 498 UpdateWindow(buttons[i]); 499 disable_but(i); 500 } 501 502 /* Creating the SLIDER window */ 503 slider = CreateWindow(TRACKBAR_CLASS, 504 TEXT(""), 505 WS_CHILD | WS_VISIBLE | TBS_NOTICKS | TBS_HORZ | TBS_ENABLESELRANGE, 506 SLIDER_CX, 507 SLIDER_CY, 508 SLIDER_W, 509 SLIDER_H, 510 hWnd, 511 (HMENU)SLIDER_ID, 512 hInst, 513 0); 514 if (!slider) 515 { 516 MessageBox(0, 0, TEXT( "CreateWindow() Error!" ), 0); 517 return FALSE; 518 } 519 520 /* Sets slider limits */ 521 SendMessage(slider, 522 TBM_SETRANGE, 523 (WPARAM)TRUE, 524 (LPARAM)MAKELONG(slider_min, slider_max)); 525 526 UpdateWindow(slider); 527 enable_but(BUTREC_ID); 528 EnableWindow(slider, FALSE); 529 break; 530 531 /* Implements slider logic */ 532 case WM_HSCROLL: 533 { 534 switch (LOWORD(wParam)) 535 { 536 case SB_ENDSCROLL: 537 break; 538 case SB_PAGERIGHT: 539 case SB_PAGELEFT: 540 case TB_THUMBTRACK: 541 /* If the user touch the slider bar, set the 542 audio start position properly */ 543 slider_pos = SendMessage(slider, TBM_GETPOS, 0, 0); 544 slid_samp = (__int64)slider_pos * (__int64)samples_max; 545 AUD_BUF->set_position(AUD_BUF->audinfo().bytes_in_samples((unsigned int)(slid_samp / (__int64)slider_max))); 546 InvalidateRect(hWnd, &text_rect, TRUE); 547 break; 548 } 549 break; 550 } 551 552 case WM_COMMAND: 553 wmId = LOWORD(wParam); 554 if ((wmId >= 0) && (wmId < 5) && (butdisabled[wmId] != FALSE)) 555 break; 556 557 switch (wmId) 558 { 559 case ID_FILE_NEW: 560 if (!isnew) 561 { 562 if (AUD_IN->current_status() == snd::WAVEIN_RECORDING) 563 AUD_IN->stop_recording(); 564 565 if ((AUD_OUT->current_status() == snd::WAVEOUT_PLAYING) || 566 (AUD_OUT->current_status() == snd::WAVEOUT_PAUSED)) 567 AUD_OUT->stop(); 568 569 AUD_BUF->reset(); 570 571 enable_but(BUTREC_ID); 572 disable_but(BUTSTART_ID); 573 disable_but(BUTEND_ID); 574 disable_but(BUTSTOP_ID); 575 disable_but(BUTPLAY_ID); 576 577 samples_max = AUD_BUF->total_samples(); 578 slider_pos = 0; 579 580 SendMessage(slider, TBM_SETPOS, (WPARAM) TRUE, (LPARAM) slider_pos); 581 582 EnableMenuItem(GetMenu(hWnd), ID_FILE_SAVEAS, MF_GRAYED); 583 EnableMenuItem(GetMenu(hWnd), ID_FILE_SAVE, MF_GRAYED); 584 585 isnew = TRUE; 586 display_dur = TRUE; 587 588 ZeroMemory(file_path, MAX_PATH * sizeof(TCHAR)); 589 590 EnableWindow(slider, FALSE); 591 592 InvalidateRect(hWnd, &text_rect, TRUE); 593 InvalidateRect(hWnd, &text2_rect, TRUE); 594 } 595 break; 596 597 case ID_FILE_OPEN: 598 ZeroMemory(&ofn, sizeof(ofn)); 599 600 ofn.lStructSize = sizeof(ofn); 601 ofn.hwndOwner = hWnd; 602 ofn.lpstrFilter = TEXT("Audio Files (*.wav)\0*.wav\0All Files (*.*)\0*.*\0"); 603 ofn.lpstrFile = file_path; 604 ofn.nMaxFile = MAX_PATH; 605 ofn.Flags = OFN_EXPLORER | OFN_FILEMUSTEXIST | OFN_HIDEREADONLY; 606 ofn.lpstrDefExt = TEXT("wav"); 607 608 if (GetOpenFileName(&ofn)) 609 { 610 open_wav(file_path); 611 EnableMenuItem(GetMenu(hWnd), ID_FILE_SAVE, MF_ENABLED); 612 EnableMenuItem(GetMenu(hWnd), ID_FILE_SAVEAS, MF_ENABLED); 613 614 EnableWindow(slider, TRUE); 615 } 616 617 InvalidateRect(hWnd, &text_rect, TRUE); 618 InvalidateRect(hWnd, &text2_rect, TRUE); 619 break; 620 621 case ID_ABOUT: 622 DialogBox(hInst, MAKEINTRESOURCE(IDD_ABOUTBOX), hWnd, AboutDlgProc); 623 return TRUE; 624 break; 625 626 case ID_FILE_SAVEAS: 627 ZeroMemory(&ofn, sizeof(ofn)); 628 629 ofn.lStructSize = sizeof(ofn); 630 ofn.hwndOwner = hWnd ; 631 ofn.Flags = OFN_OVERWRITEPROMPT; 632 ofn.lpstrFilter = TEXT("Audio Files (*.wav)\0*.wav\0All Files (*.*)\0*.*\0"); 633 ofn.lpstrFile = file_path; 634 ofn.nMaxFile = MAX_PATH; 635 636 ofn.lpstrDefExt = TEXT("wav"); 637 638 if (GetSaveFileName (&ofn)) 639 { 640 write_wav(file_path); 641 EnableMenuItem(GetMenu(hWnd), ID_FILE_SAVE, MF_ENABLED); 642 } 643 break; 644 645 646 case ID_EDIT_AUDIOPROPS: 647 ShellExecute(NULL, NULL, _T("rundll32.exe"), _T("shell32.dll,Control_RunDLL mmsys.cpl,ShowAudioPropertySheet"), NULL, SW_SHOWNORMAL); 648 break; 649 650 case ID_FILE_EXIT: 651 DestroyWindow(hWnd); 652 break; 653 654 /* Sndrec32 buttons routines */ 655 case BUTSTART_ID: 656 AUD_BUF->set_position_start(); 657 slider_pos = 0; 658 SendMessage(slider, TBM_SETPOS, (WPARAM)TRUE, (LPARAM)slider_pos); 659 break; 660 661 case BUTEND_ID: 662 DestroyWindow(hWnd); 663 break; 664 665 case BUTPLAY_ID: 666 if (wout_first == false) 667 { 668 AUD_OUT->open(); 669 wout_first = true; 670 } 671 672 AUD_OUT->play(); 673 674 disable_but(BUTSTART_ID); 675 disable_but(BUTEND_ID); 676 disable_but(BUTREC_ID); 677 disable_but(BUTPLAY_ID); 678 679 SetTimer(hWnd, 1, 250, 0); 680 SetTimer(hWnd, WAVEBAR_TIMERID, WAVEBAR_TIMERTIME, 0); 681 682 break; 683 684 case BUTSTOP_ID: 685 if (s_recording) 686 { 687 s_recording = FALSE; 688 689 AUD_IN->stop_recording(); 690 691 /* Resetting slider position */ 692 slider_pos = 0; 693 SendMessage(slider, TBM_SETPOS, (WPARAM)TRUE, (LPARAM)slider_pos); 694 695 samples_max = AUD_BUF->samples_received(); 696 697 EnableMenuItem((HMENU)IDR_MENU1, ID_FILE_SAVEAS, MF_ENABLED); 698 699 enable_but(BUTSTART_ID); 700 enable_but(BUTEND_ID); 701 enable_but(BUTREC_ID); 702 enable_but(BUTPLAY_ID); 703 704 EnableMenuItem(GetMenu(hWnd), ID_FILE_SAVEAS, MF_ENABLED); 705 EnableWindow(slider, TRUE); 706 707 display_dur = FALSE; 708 709 AUD_BUF->truncate(); 710 711 InvalidateRect(hWnd, &text_rect, TRUE); 712 InvalidateRect(wave_win, 0, TRUE); 713 714 } 715 else 716 { 717 AUD_OUT->pause(); 718 719 enable_but(BUTSTART_ID); 720 enable_but(BUTEND_ID); 721 enable_but(BUTREC_ID); 722 enable_but(BUTPLAY_ID); 723 724 } 725 726 KillTimer(hWnd, 1); 727 KillTimer(hWnd, WAVEBAR_TIMERID); 728 729 InvalidateRect(hWnd, &text_rect, TRUE); 730 731 break; 732 733 case BUTREC_ID: 734 if (win_first == false) 735 { 736 AUD_IN->open(); 737 win_first = true; 738 } 739 740 s_recording = TRUE; 741 742 samples_max = AUD_BUF->total_samples(); 743 744 AUD_IN->start_recording(); 745 746 enable_but(BUTSTOP_ID); 747 748 disable_but(BUTSTART_ID); 749 disable_but(BUTEND_ID); 750 disable_but(BUTREC_ID); 751 disable_but(BUTPLAY_ID); 752 753 isnew = FALSE; 754 755 EnableWindow(slider, FALSE); 756 757 SetTimer(hWnd, 1, 150, 0); 758 SetTimer(hWnd, WAVEBAR_TIMERID, WAVEBAR_TIMERTIME, 0); 759 760 break; 761 762 default: 763 return DefWindowProc(hWnd, message, wParam, lParam); 764 } 765 break; 766 767 case WM_TIMER: 768 switch (wParam) 769 { 770 case 1: 771 if (stopped_flag) 772 { 773 KillTimer(hWnd, 1); 774 KillTimer(hWnd, WAVEBAR_TIMERID); 775 slider_pos = 0; 776 enable_but(BUTPLAY_ID); 777 stopped_flag = FALSE; 778 } 779 780 SendMessage(slider, TBM_SETPOS, (WPARAM)TRUE, (LPARAM)slider_pos); 781 InvalidateRect(hWnd, &text_rect, TRUE); 782 break; 783 784 case WAVEBAR_TIMERID: 785 InvalidateRect(wave_win, 0, TRUE); 786 SendMessage(wave_win, WM_USER, 0, 0); 787 break; 788 } 789 break; 790 791 case WM_PAINT: 792 hdc = BeginPaint(hWnd, &ps); 793 font = CreateFontIndirect(&s_info.lfMenuFont); 794 oldfont = (HFONT) SelectObject(hdc, font); 795 SetBkMode(hdc, TRANSPARENT); 796 797 if (AUD_IN->current_status() == snd::WAVEIN_RECORDING) 798 { 799 gprintf(str_tmp, 800 MAX_LOADSTRING, 801 str_pos, 802 (float)((float)AUD_BUF->bytes_recorded() / (float)AUD_BUF->audinfo().byte_rate())); 803 804 } 805 else if (AUD_OUT->current_status() == snd::WAVEOUT_PLAYING) 806 { 807 gprintf(str_tmp, 808 MAX_LOADSTRING, 809 str_pos, 810 (float)((float)AUD_BUF->bytes_played() / (float)AUD_BUF->audinfo().byte_rate())); 811 } 812 else 813 { 814 gprintf(str_tmp, 815 MAX_LOADSTRING, 816 str_pos, 817 (float)((((float)slider_pos * (float)samples_max) / (float)slider_max) / (float)AUD_BUF->audinfo().sample_rate())); 818 } 819 820 ExtTextOut(hdc, 821 STRPOS_X, 822 STRPOS_Y, 823 ETO_OPAQUE, 824 0, 825 str_tmp, 826 _tcslen(str_tmp), 827 0); 828 829 if (display_dur) 830 { 831 gprintf(str_tmp, 832 MAX_LOADSTRING, 833 str_dur, 834 AUD_BUF->fseconds_total()); 835 } 836 else 837 { 838 gprintf(str_tmp, 839 MAX_LOADSTRING, 840 str_dur, 841 AUD_BUF->fseconds_recorded()); 842 } 843 844 ExtTextOut(hdc, 845 STRDUR_X, 846 STRDUR_Y, 847 ETO_OPAQUE, 848 0, 849 str_tmp, 850 _tcslen(str_tmp), 851 0); 852 853 gprintf(str_tmp, 854 MAX_LOADSTRING, 855 str_buf, 856 (float)((float)AUD_BUF->mem_size() / 1024.0f)); 857 858 ExtTextOut(hdc, 859 STRBUF_X, 860 STRBUF_Y, 861 ETO_OPAQUE, 862 0, 863 str_tmp, 864 _tcslen(str_tmp), 865 0); 866 867 gprintf(str_tmp, 868 MAX_LOADSTRING, 869 str_fmt, 870 (float)((float)AUD_BUF->audinfo().sample_rate() / 1000.0f), 871 AUD_BUF->audinfo().bits(), 872 AUD_BUF->audinfo().channels() == 2 ? str_mono : str_stereo); 873 874 ExtTextOut(hdc, 875 STRFMT_X, 876 STRFMT_Y, 877 ETO_OPAQUE, 878 0, 879 str_tmp, 880 _tcslen(str_tmp), 881 0); 882 883 gprintf(str_tmp, 884 MAX_LOADSTRING, 885 str_chan, 886 AUD_BUF->audinfo().channels() == 2 ? str_stereo : str_mono); 887 888 ExtTextOut(hdc, 889 STRCHAN_X, 890 STRCHAN_Y, 891 ETO_OPAQUE, 892 0, 893 str_tmp, 894 _tcslen(str_tmp), 895 0); 896 897 SelectObject(hdc, oldfont); 898 DeleteObject(font); 899 EndPaint(hWnd, &ps); 900 break; 901 902 case WM_DESTROY: 903 PostQuitMessage(0); 904 break; 905 default: 906 return DefWindowProc(hWnd, message, wParam, lParam); 907 } 908 909 return 0; 910 } 911 912 void l_play_finished(void) 913 { 914 stopped_flag = true; 915 916 enable_but(BUTSTART_ID); 917 enable_but(BUTEND_ID); 918 enable_but(BUTREC_ID); 919 enable_but(BUTPLAY_ID); 920 921 InvalidateRect(wave_win, 0, TRUE); 922 } 923 924 void l_audio_arrival(unsigned int samples_arrival) 925 { 926 slider_pos += (DWORD)((slider_max * samples_arrival) / samples_max); 927 } 928 929 void l_buffer_resized(unsigned int new_size) 930 { 931 } 932 933 VOID enable_but(DWORD id) 934 { 935 butdisabled[id] = FALSE; 936 SendMessage(buttons[id], BM_SETIMAGE, (WPARAM)IMAGE_BITMAP, (LPARAM)butbmps[id]); 937 } 938 939 VOID disable_but(DWORD id) 940 { 941 butdisabled[id] = TRUE; 942 SendMessage(buttons[id], BM_SETIMAGE, (WPARAM)IMAGE_BITMAP, (LPARAM)butbmps_dis[id]); 943 } 944 945 BOOL open_wav(TCHAR *f) 946 { 947 HANDLE file; 948 949 riff_hdr r; 950 wave_hdr w; 951 data_chunk d; 952 953 BOOL b; 954 955 DWORD bytes_recorded_in_wav = 0; 956 DWORD is_read = 0; 957 958 file = CreateFile(f, 959 GENERIC_READ, 960 0, 961 0, 962 OPEN_EXISTING, 963 FILE_ATTRIBUTE_NORMAL, 964 0); 965 if (!file) 966 { 967 MessageBox(main_win, 968 TEXT("Cannot open file. CreateFile() error."), 969 TEXT("ERROR"), 970 MB_OK | MB_ICONERROR); 971 972 return FALSE; 973 } 974 975 b = ReadFile(file, (LPVOID)&r, sizeof(r), &is_read, 0); 976 if (!b) 977 { 978 MessageBox(main_win, 979 TEXT("Cannot read RIFF header."), 980 TEXT("ERROR"), 981 MB_OK | MB_ICONERROR); 982 983 CloseHandle(file); 984 return FALSE; 985 } 986 987 b = ReadFile(file, (LPVOID)&w, sizeof(w), &is_read, 0); 988 if (!b) 989 { 990 MessageBox(main_win, 991 TEXT("Cannot read WAVE header."), 992 TEXT("ERROR"), 993 MB_OK | MB_ICONERROR); 994 995 CloseHandle(file); 996 return FALSE; 997 } 998 999 b = ReadFile(file, (LPVOID)&d, sizeof(d), &is_read, 0); 1000 if (!b) 1001 { 1002 MessageBox(main_win, 1003 TEXT("Cannot read WAVE subchunk."), 1004 TEXT("ERROR"), 1005 MB_OK | MB_ICONERROR); 1006 1007 CloseHandle(file); 1008 return FALSE; 1009 } 1010 1011 bytes_recorded_in_wav = r.chunksize - 36; 1012 if (bytes_recorded_in_wav == 0) 1013 { 1014 MessageBox(main_win, 1015 TEXT("Cannot read file. No audio data."), 1016 TEXT("ERROR"), 1017 MB_OK | MB_ICONERROR); 1018 1019 CloseHandle(file); 1020 return FALSE; 1021 } 1022 1023 snd::audio_format openfmt(w.SampleRate, w.BitsPerSample, w.NumChannels); 1024 1025 AUD_BUF->clear(); 1026 AUD_BUF->alloc_bytes(bytes_recorded_in_wav); 1027 1028 b = ReadFile(file, 1029 (LPVOID)AUD_BUF->audio_buffer(), 1030 bytes_recorded_in_wav, 1031 &is_read, 1032 0); 1033 1034 AUD_BUF->set_b_received(bytes_recorded_in_wav); 1035 1036 if ((!b) || (is_read != bytes_recorded_in_wav)) 1037 { 1038 MessageBox(main_win, 1039 TEXT("Cannot read file. Error reading audio data."), 1040 TEXT("ERROR"), 1041 MB_OK | MB_ICONERROR); 1042 1043 CloseHandle(file); 1044 1045 AUD_BUF->reset(); 1046 return FALSE; 1047 } 1048 1049 CloseHandle(file); 1050 1051 enable_but(BUTPLAY_ID); 1052 enable_but(BUTSTOP_ID); 1053 enable_but(BUTSTART_ID); 1054 enable_but(BUTEND_ID); 1055 enable_but(BUTREC_ID); 1056 1057 samples_max = AUD_BUF->samples_received(); 1058 1059 isnew = FALSE; 1060 1061 return TRUE; 1062 } 1063 1064 BOOL 1065 write_wav(TCHAR *f) 1066 { 1067 HANDLE file; 1068 DWORD written; 1069 BOOL is_writ; 1070 int i; 1071 riff_hdr r; 1072 wave_hdr w; 1073 data_chunk d; 1074 1075 file = CreateFile(f, GENERIC_WRITE, 0, 0, CREATE_NEW, FILE_ATTRIBUTE_NORMAL, 0); 1076 if (!file) 1077 { 1078 i = MessageBox(main_win, 1079 TEXT("File already exist. Overwrite it?"), 1080 TEXT("Warning"), 1081 MB_YESNO | MB_ICONQUESTION); 1082 1083 if (i == IDYES) 1084 { 1085 file = CreateFile(f, GENERIC_WRITE, 0, 0, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, 0); 1086 if (!file) 1087 { 1088 MessageBox(main_win, 1089 TEXT("File Error, CreateFile() failed."), 1090 TEXT("ERROR"), 1091 MB_OK | MB_ICONERROR); 1092 1093 return FALSE; 1094 } 1095 1096 } else 1097 return FALSE; 1098 } 1099 1100 r.magic = 0x46464952; 1101 r.format = 0x45564157; 1102 r.chunksize = 36 + AUD_BUF->bytes_recorded(); 1103 1104 w.Subchunkid = 0x20746d66; 1105 w.Subchunk1Size = 16; 1106 w.AudioFormat = 1; 1107 w.NumChannels = AUD_BUF->audinfo().channels(); 1108 w.SampleRate = AUD_BUF->audinfo().sample_rate(); 1109 w.ByteRate = AUD_BUF->audinfo().byte_rate(); 1110 w.BlockAlign = AUD_BUF->audinfo().block_align(); 1111 w.BitsPerSample = AUD_BUF->audinfo().bits(); 1112 1113 d.subc = 0x61746164; 1114 d.subc_size = AUD_BUF->bytes_recorded(); 1115 1116 /* Writing headers */ 1117 is_writ = WriteFile(file, (LPCVOID)&r, sizeof(r), &written, 0); 1118 if (!is_writ) 1119 { 1120 MessageBox(main_win, 1121 TEXT("File Error, WriteFile() failed."), 1122 TEXT("ERROR"), 1123 MB_OK | MB_ICONERROR); 1124 1125 CloseHandle(file); 1126 return FALSE; 1127 } 1128 1129 is_writ = WriteFile(file, (LPCVOID)&w, sizeof(w), &written, 0); 1130 if (!is_writ) 1131 { 1132 MessageBox(main_win, 1133 TEXT("File Error, WriteFile() failed."), 1134 TEXT("ERROR"), 1135 MB_OK | MB_ICONERROR); 1136 1137 CloseHandle(file); 1138 return FALSE; 1139 } 1140 1141 is_writ = WriteFile(file, (LPCVOID)&d, sizeof(d), &written, 0); 1142 if (!is_writ) 1143 { 1144 MessageBox(main_win, 1145 TEXT("File Error, WriteFile() failed."), 1146 TEXT("ERROR"), 1147 MB_OK | MB_ICONERROR); 1148 1149 CloseHandle(file); 1150 return FALSE; 1151 } 1152 1153 is_writ = WriteFile(file, 1154 (LPCVOID)AUD_BUF->audio_buffer(), 1155 AUD_BUF->bytes_recorded(), 1156 &written, 1157 0); 1158 if (!is_writ) 1159 { 1160 MessageBox(main_win, 1161 TEXT("File Error, WriteFile() failed."), 1162 TEXT("ERROR"), 1163 MB_OK | MB_ICONERROR); 1164 1165 CloseHandle(file); 1166 return FALSE; 1167 } 1168 1169 CloseHandle(file); 1170 return TRUE; 1171 } 1172