157e2f323SJérôme Duval /* 257e2f323SJérôme Duval * Copyright 2005, Jérôme Duval. All rights reserved. 357e2f323SJérôme Duval * Distributed under the terms of the MIT License. 457e2f323SJérôme Duval * 5ad0c65eaSJérôme Duval * Inspired by SoundCapture from Be newsletter (Media Kit Basics: 6ad0c65eaSJérôme Duval * Consumers and Producers) 757e2f323SJérôme Duval */ 857e2f323SJérôme Duval 957e2f323SJérôme Duval #include <Application.h> 1057e2f323SJérôme Duval #include <Alert.h> 1157e2f323SJérôme Duval #include <Debug.h> 1257e2f323SJérôme Duval #include <Screen.h> 1357e2f323SJérôme Duval #include <Button.h> 1457e2f323SJérôme Duval #include <CheckBox.h> 1557e2f323SJérôme Duval #include <TextControl.h> 1657e2f323SJérôme Duval #include <MenuField.h> 1757e2f323SJérôme Duval #include <PopUpMenu.h> 1857e2f323SJérôme Duval #include <MenuItem.h> 1957e2f323SJérôme Duval #include <Box.h> 2057e2f323SJérôme Duval #include <ScrollView.h> 2157e2f323SJérôme Duval #include <Beep.h> 2257e2f323SJérôme Duval #include <StringView.h> 2357e2f323SJérôme Duval #include <String.h> 2457e2f323SJérôme Duval #include <Slider.h> 2557e2f323SJérôme Duval #include <Message.h> 2657e2f323SJérôme Duval 2757e2f323SJérôme Duval #include <Path.h> 2857e2f323SJérôme Duval #include <FindDirectory.h> 2957e2f323SJérôme Duval #include <MediaAddOn.h> 3057e2f323SJérôme Duval 3157e2f323SJérôme Duval #include <SoundPlayer.h> 3257e2f323SJérôme Duval 3357e2f323SJérôme Duval #include <assert.h> 3457e2f323SJérôme Duval #include <stdio.h> 3557e2f323SJérôme Duval #include <string.h> 3657e2f323SJérôme Duval #include <stdlib.h> 3757e2f323SJérôme Duval #include <ctype.h> 3857e2f323SJérôme Duval #include <unistd.h> 39338b8dc3SIngo Weinhold #include <fcntl.h> 4057e2f323SJérôme Duval 4157e2f323SJérôme Duval #include <MediaRoster.h> 4257e2f323SJérôme Duval #include <TimeSource.h> 43019ed09bSJérôme Duval #include <NodeInfo.h> 4457e2f323SJérôme Duval 4557e2f323SJérôme Duval #include "RecorderWindow.h" 4657e2f323SJérôme Duval #include "SoundConsumer.h" 4757e2f323SJérôme Duval #include "FileUtils.h" 4857e2f323SJérôme Duval 4957e2f323SJérôme Duval #if ! NDEBUG 5057e2f323SJérôme Duval #define FPRINTF(args) fprintf args 5157e2f323SJérôme Duval #else 5257e2f323SJérôme Duval #define FPRINTF(args) 5357e2f323SJérôme Duval #endif 5457e2f323SJérôme Duval 5557e2f323SJérôme Duval #define DEATH FPRINTF 5657e2f323SJérôme Duval #define CONNECT FPRINTF 5757e2f323SJérôme Duval #define WINDOW FPRINTF 5857e2f323SJérôme Duval 59546208a5SOliver Tappe #undef B_TRANSLATION_CONTEXT 60546208a5SOliver Tappe #define B_TRANSLATION_CONTEXT "RecorderWindow" 613ee96407SJérôme Duval 623ee96407SJérôme Duval 6357e2f323SJérôme Duval // default window positioning 6457e2f323SJérôme Duval static const float MIN_WIDTH = 400.0f; 65b271ff3fSPhilippe Saint-Pierre static const float MIN_HEIGHT = 175.0f; 6657e2f323SJérôme Duval static const float XPOS = 100.0f; 6757e2f323SJérôme Duval static const float YPOS = 200.0f; 6857e2f323SJérôme Duval 69ad0c65eaSJérôme Duval #define FOURCC(a,b,c,d) ((((uint32)(d)) << 24) | (((uint32)(c)) << 16) \ 70ad0c65eaSJérôme Duval | (((uint32)(b)) << 8) | ((uint32)(a))) 7157e2f323SJérôme Duval 7257e2f323SJérôme Duval struct riff_struct 7357e2f323SJérôme Duval { 7457e2f323SJérôme Duval uint32 riff_id; // 'RIFF' 7557e2f323SJérôme Duval uint32 len; 7657e2f323SJérôme Duval uint32 wave_id; // 'WAVE' 7757e2f323SJérôme Duval }; 7857e2f323SJérôme Duval 7957e2f323SJérôme Duval struct chunk_struct 8057e2f323SJérôme Duval { 8157e2f323SJérôme Duval uint32 fourcc; 8257e2f323SJérôme Duval uint32 len; 8357e2f323SJérôme Duval }; 8457e2f323SJérôme Duval 8557e2f323SJérôme Duval struct format_struct 8657e2f323SJérôme Duval { 8757e2f323SJérôme Duval uint16 format_tag; 8857e2f323SJérôme Duval uint16 channels; 8957e2f323SJérôme Duval uint32 samples_per_sec; 9057e2f323SJérôme Duval uint32 avg_bytes_per_sec; 9157e2f323SJérôme Duval uint16 block_align; 9257e2f323SJérôme Duval uint16 bits_per_sample; 9357e2f323SJérôme Duval }; 9457e2f323SJérôme Duval 9557e2f323SJérôme Duval 9657e2f323SJérôme Duval struct wave_struct 9757e2f323SJérôme Duval { 9857e2f323SJérôme Duval struct riff_struct riff; 9957e2f323SJérôme Duval struct chunk_struct format_chunk; 10057e2f323SJérôme Duval struct format_struct format; 10157e2f323SJérôme Duval struct chunk_struct data_chunk; 10257e2f323SJérôme Duval }; 10357e2f323SJérôme Duval 10457e2f323SJérôme Duval 10571302e7cSJonas Sundström RecorderWindow::RecorderWindow() 10671302e7cSJonas Sundström : 10771302e7cSJonas Sundström BWindow(BRect(XPOS, YPOS, XPOS + MIN_WIDTH, YPOS + MIN_HEIGHT), 108560ff447SJonas Sundström B_TRANSLATE_SYSTEM_NAME("SoundRecorder"), B_TITLED_WINDOW, 10971302e7cSJonas Sundström B_ASYNCHRONOUS_CONTROLS | B_NOT_V_RESIZABLE | B_NOT_ZOOMABLE), 11057e2f323SJérôme Duval fPlayer(NULL), 1115fa77532SJérôme Duval fSoundList(NULL), 11257e2f323SJérôme Duval fPlayFile(NULL), 11357e2f323SJérôme Duval fPlayTrack(NULL), 1147942339dSJérôme Duval fPlayFrames(0), 115948356deSJérôme Duval fLooping(false), 1165fa77532SJérôme Duval fSavePanel(NULL), 117cff7eda5SWim van der Meer fInitCheck(B_OK) 11857e2f323SJérôme Duval { 11957e2f323SJérôme Duval fRoster = NULL; 12057e2f323SJérôme Duval fRecordButton = NULL; 12157e2f323SJérôme Duval fPlayButton = NULL; 12257e2f323SJérôme Duval fStopButton = NULL; 12357e2f323SJérôme Duval fSaveButton = NULL; 124948356deSJérôme Duval fLoopButton = NULL; 12557e2f323SJérôme Duval fInputField = NULL; 12657e2f323SJérôme Duval fRecordNode = 0; 12757e2f323SJérôme Duval fRecording = false; 12857e2f323SJérôme Duval fTempCount = -1; 12957e2f323SJérôme Duval fButtonState = btnPaused; 13057e2f323SJérôme Duval 13157e2f323SJérôme Duval CalcSizes(MIN_WIDTH, MIN_HEIGHT); 13257e2f323SJérôme Duval 1335fa77532SJérôme Duval fInitCheck = InitWindow(); 1345fa77532SJérôme Duval if (fInitCheck != B_OK) { 135bdb1d3acSJérôme Duval if (fInitCheck == B_NAME_NOT_FOUND) 1363ee96407SJérôme Duval ErrorAlert(B_TRANSLATE("Cannot find default audio hardware"), 1373ee96407SJérôme Duval fInitCheck); 138bdb1d3acSJérôme Duval else 1393ee96407SJérôme Duval ErrorAlert(B_TRANSLATE("Cannot connect to media server"), 1403ee96407SJérôme Duval fInitCheck); 1415fa77532SJérôme Duval PostMessage(B_QUIT_REQUESTED); 1425fa77532SJérôme Duval } else 1435fa77532SJérôme Duval Show(); 14457e2f323SJérôme Duval } 14557e2f323SJérôme Duval 1468e32a27eSPhilippe Saint-Pierre 14757e2f323SJérôme Duval RecorderWindow::~RecorderWindow() 14857e2f323SJérôme Duval { 149ad0c65eaSJérôme Duval // The sound consumer and producer are Nodes; it has to be released and the Roster 15057e2f323SJérôme Duval // will reap it when it's done. 151b1ed9a63SJérôme Duval if (fRecordNode) 15257e2f323SJérôme Duval fRecordNode->Release(); 15357e2f323SJérôme Duval delete fPlayer; 1547942339dSJérôme Duval 1557942339dSJérôme Duval if (fPlayTrack && fPlayFile) 1567942339dSJérôme Duval fPlayFile->ReleaseTrack(fPlayTrack); 1577942339dSJérôme Duval if (fPlayFile) 1587942339dSJérôme Duval delete fPlayFile; 1597942339dSJérôme Duval fPlayTrack = NULL; 1607942339dSJérôme Duval fPlayFile = NULL; 1617942339dSJérôme Duval 16257e2f323SJérôme Duval // Clean up items in list view. 16357e2f323SJérôme Duval if (fSoundList) { 16457e2f323SJérôme Duval fSoundList->DeselectAll(); 165ad0c65eaSJérôme Duval for (int i = 0; i < fSoundList->CountItems(); i++) { 166ad0c65eaSJérôme Duval WINDOW((stderr, "clean up item %d\n", i+1)); 167ad0c65eaSJérôme Duval SoundListItem* item = dynamic_cast<SoundListItem *>(fSoundList->ItemAt(i)); 16857e2f323SJérôme Duval if (item) { 169b1ed9a63SJérôme Duval if (item->IsTemp()) 17057e2f323SJérôme Duval item->Entry().Remove(); // delete temp file 17157e2f323SJérôme Duval delete item; 17257e2f323SJérôme Duval } 17357e2f323SJérôme Duval } 17457e2f323SJérôme Duval fSoundList->MakeEmpty(); 17557e2f323SJérôme Duval } 17657e2f323SJérôme Duval // Clean up currently recording file, if any. 17757e2f323SJérôme Duval fRecEntry.Remove(); 17857e2f323SJérôme Duval fRecEntry.Unset(); 17905ea8535SKarsten Heimrich 18005ea8535SKarsten Heimrich delete fSavePanel; 18157e2f323SJérôme Duval } 18257e2f323SJérôme Duval 18357e2f323SJérôme Duval 1845fa77532SJérôme Duval status_t 1855fa77532SJérôme Duval RecorderWindow::InitCheck() 1865fa77532SJérôme Duval { 1875fa77532SJérôme Duval return fInitCheck; 1885fa77532SJérôme Duval } 1895fa77532SJérôme Duval 1905fa77532SJérôme Duval 19157e2f323SJérôme Duval void 192b1ed9a63SJérôme Duval RecorderWindow::CalcSizes(float min_width, float min_height) 19357e2f323SJérôme Duval { 19457e2f323SJérôme Duval // Set up size limits based on new screen size 1955e99b7dfSJérôme Duval BScreen screen(this); 196b1ed9a63SJérôme Duval BRect rect = screen.Frame(); 197b1ed9a63SJérôme Duval float width = rect.Width() - 12; 198b1ed9a63SJérôme Duval SetSizeLimits(min_width, width, min_height, rect.Height() - 24); 19957e2f323SJérôme Duval 20057e2f323SJérôme Duval // Don't zoom to cover all of screen; user can resize last bit if necessary. 20157e2f323SJérôme Duval // This leaves other windows visible. 202b1ed9a63SJérôme Duval if (width > 640) 203b1ed9a63SJérôme Duval width = 640 + (width - 640) / 2; 204b1ed9a63SJérôme Duval SetZoomLimits(width, rect.Height() - 24); 20557e2f323SJérôme Duval } 20657e2f323SJérôme Duval 20757e2f323SJérôme Duval 20857e2f323SJérôme Duval status_t 20957e2f323SJérôme Duval RecorderWindow::InitWindow() 21057e2f323SJérôme Duval { 21157e2f323SJérôme Duval BPopUpMenu * popup = 0; 21257e2f323SJérôme Duval status_t error; 21357e2f323SJérôme Duval 21457e2f323SJérôme Duval try { 21557e2f323SJérôme Duval // Find temp directory for recorded sounds. 21657e2f323SJérôme Duval BPath path; 217b1ed9a63SJérôme Duval if (!(error = find_directory(B_COMMON_TEMP_DIRECTORY, &path))) 21857e2f323SJérôme Duval error = fTempDir.SetTo(path.Path()); 219b1ed9a63SJérôme Duval if (error < 0) 22057e2f323SJérôme Duval goto bad_mojo; 22157e2f323SJérôme Duval 22257e2f323SJérôme Duval // Make sure the media roster is there (which means the server is there). 22357e2f323SJérôme Duval fRoster = BMediaRoster::Roster(&error); 224b1ed9a63SJérôme Duval if (!fRoster) 22557e2f323SJérôme Duval goto bad_mojo; 22657e2f323SJérôme Duval 22757e2f323SJérôme Duval error = fRoster->GetAudioInput(&fAudioInputNode); 228b1ed9a63SJérôme Duval if (error < B_OK) // there's no input? 22957e2f323SJérôme Duval goto bad_mojo; 23057e2f323SJérôme Duval 23157e2f323SJérôme Duval error = fRoster->GetAudioMixer(&fAudioMixerNode); 232b1ed9a63SJérôme Duval if (error < B_OK) // there's no mixer? 23357e2f323SJérôme Duval goto bad_mojo; 23457e2f323SJérôme Duval 23557e2f323SJérôme Duval // Create our internal Node which records sound, and register it. 23657e2f323SJérôme Duval fRecordNode = new SoundConsumer("Sound Recorder"); 23757e2f323SJérôme Duval error = fRoster->RegisterNode(fRecordNode); 238b1ed9a63SJérôme Duval if (error < B_OK) 23957e2f323SJérôme Duval goto bad_mojo; 24057e2f323SJérôme Duval 24157e2f323SJérôme Duval // Create the window header with controls 24257e2f323SJérôme Duval BRect r(Bounds()); 24357e2f323SJérôme Duval r.bottom = r.top + 175; 244ad0c65eaSJérôme Duval BBox *background = new BBox(r, "_background", B_FOLLOW_LEFT_RIGHT 245ad0c65eaSJérôme Duval | B_FOLLOW_TOP, B_WILL_DRAW | B_FRAME_EVENTS | B_NAVIGABLE_JUMP, B_NO_BORDER); 24657e2f323SJérôme Duval AddChild(background); 24757e2f323SJérôme Duval 24857e2f323SJérôme Duval r = background->Bounds(); 249a4797804SJérôme Duval r.left = 0; 2508881ad65SJérôme Duval r.right = r.left + 38; 25157e2f323SJérôme Duval r.bottom = r.top + 104; 25257e2f323SJérôme Duval fVUView = new VUView(r, B_FOLLOW_LEFT|B_FOLLOW_TOP); 25357e2f323SJérôme Duval background->AddChild(fVUView); 25457e2f323SJérôme Duval 25557e2f323SJérôme Duval r = background->Bounds(); 2568881ad65SJérôme Duval r.left = r.left + 40; 25757e2f323SJérôme Duval r.bottom = r.top + 104; 25857e2f323SJérôme Duval fScopeView = new ScopeView(r, B_FOLLOW_LEFT_RIGHT|B_FOLLOW_TOP); 25957e2f323SJérôme Duval background->AddChild(fScopeView); 26057e2f323SJérôme Duval 26157e2f323SJérôme Duval r = background->Bounds(); 26257e2f323SJérôme Duval r.left = 2; 26357e2f323SJérôme Duval r.right -= 26; 26457e2f323SJérôme Duval r.top = 115; 26557e2f323SJérôme Duval r.bottom = r.top + 30; 266ad0c65eaSJérôme Duval fTrackSlider = new TrackSlider(r, "trackSlider", new BMessage(POSITION_CHANGED), 267ad0c65eaSJérôme Duval B_FOLLOW_LEFT_RIGHT | B_FOLLOW_TOP); 26857e2f323SJérôme Duval background->AddChild(fTrackSlider); 26957e2f323SJérôme Duval 27057e2f323SJérôme Duval BRect buttonRect; 27157e2f323SJérôme Duval 27257e2f323SJérôme Duval // Button for rewinding 27357e2f323SJérôme Duval buttonRect = BRect(BPoint(0,0), kSkipButtonSize); 27457e2f323SJérôme Duval buttonRect.OffsetTo(background->Bounds().LeftBottom() - BPoint(-7, 25)); 2753ee96407SJérôme Duval fRewindButton = new TransportButton(buttonRect, B_TRANSLATE("Rewind"), 276ad0c65eaSJérôme Duval kSkipBackBitmapBits, kPressedSkipBackBitmapBits, 277ad0c65eaSJérôme Duval kDisabledSkipBackBitmapBits, new BMessage(REWIND)); 27857e2f323SJérôme Duval background->AddChild(fRewindButton); 27957e2f323SJérôme Duval 28057e2f323SJérôme Duval // Button for stopping recording or playback 28157e2f323SJérôme Duval buttonRect = BRect(BPoint(0,0), kStopButtonSize); 28257e2f323SJérôme Duval buttonRect.OffsetTo(background->Bounds().LeftBottom() - BPoint(-48, 25)); 2833ee96407SJérôme Duval fStopButton = new TransportButton(buttonRect, B_TRANSLATE("Stop"), 284ad0c65eaSJérôme Duval kStopButtonBitmapBits, kPressedStopButtonBitmapBits, 285ad0c65eaSJérôme Duval kDisabledStopButtonBitmapBits, new BMessage(STOP)); 28657e2f323SJérôme Duval background->AddChild(fStopButton); 28757e2f323SJérôme Duval 28857e2f323SJérôme Duval // Button for starting playback of selected sound 28957e2f323SJérôme Duval BRect playRect(BPoint(0,0), kPlayButtonSize); 29057e2f323SJérôme Duval playRect.OffsetTo(background->Bounds().LeftBottom() - BPoint(-82, 25)); 2913ee96407SJérôme Duval fPlayButton = new PlayPauseButton(playRect, B_TRANSLATE("Play"), 29257e2f323SJérôme Duval new BMessage(PLAY), new BMessage(PLAY_PERIOD), ' ', 0); 29357e2f323SJérôme Duval background->AddChild(fPlayButton); 29457e2f323SJérôme Duval 29557e2f323SJérôme Duval // Button for forwarding 29657e2f323SJérôme Duval buttonRect = BRect(BPoint(0,0), kSkipButtonSize); 29757e2f323SJérôme Duval buttonRect.OffsetTo(background->Bounds().LeftBottom() - BPoint(-133, 25)); 2983ee96407SJérôme Duval fForwardButton = new TransportButton(buttonRect, B_TRANSLATE("Forward"), 299ad0c65eaSJérôme Duval kSkipForwardBitmapBits, kPressedSkipForwardBitmapBits, 300ad0c65eaSJérôme Duval kDisabledSkipForwardBitmapBits, new BMessage(FORWARD)); 30157e2f323SJérôme Duval background->AddChild(fForwardButton); 30257e2f323SJérôme Duval 30357e2f323SJérôme Duval // Button to start recording (or waiting for sound) 30457e2f323SJérôme Duval buttonRect = BRect(BPoint(0,0), kRecordButtonSize); 30557e2f323SJérôme Duval buttonRect.OffsetTo(background->Bounds().LeftBottom() - BPoint(-174, 25)); 3063ee96407SJérôme Duval fRecordButton = new RecordButton(buttonRect, B_TRANSLATE("Record"), 30757e2f323SJérôme Duval new BMessage(RECORD), new BMessage(RECORD_PERIOD)); 30857e2f323SJérôme Duval background->AddChild(fRecordButton); 30957e2f323SJérôme Duval 31057e2f323SJérôme Duval // Button for saving selected sound 31157e2f323SJérôme Duval buttonRect = BRect(BPoint(0,0), kDiskButtonSize); 31257e2f323SJérôme Duval buttonRect.OffsetTo(background->Bounds().LeftBottom() - BPoint(-250, 21)); 3133ee96407SJérôme Duval fSaveButton = new TransportButton(buttonRect, B_TRANSLATE("Save"), 314ad0c65eaSJérôme Duval kDiskButtonBitmapsBits, kPressedDiskButtonBitmapsBits, 315ad0c65eaSJérôme Duval kDisabledDiskButtonBitmapsBits, new BMessage(SAVE)); 31657e2f323SJérôme Duval fSaveButton->SetResizingMode(B_FOLLOW_RIGHT | B_FOLLOW_TOP); 31757e2f323SJérôme Duval background->AddChild(fSaveButton); 31857e2f323SJérôme Duval 319948356deSJérôme Duval // Button Loop 320948356deSJérôme Duval buttonRect = BRect(BPoint(0,0), kArrowSize); 321948356deSJérôme Duval buttonRect.OffsetTo(background->Bounds().RightBottom() - BPoint(23, 48)); 3223ee96407SJérôme Duval fLoopButton = new DrawButton(buttonRect, B_TRANSLATE("Loop"), 323948356deSJérôme Duval kLoopArrowBits, kArrowBits, new BMessage(LOOP)); 324948356deSJérôme Duval fLoopButton->SetResizingMode(B_FOLLOW_RIGHT | B_FOLLOW_TOP); 325948356deSJérôme Duval fLoopButton->SetTarget(this); 326948356deSJérôme Duval background->AddChild(fLoopButton); 327948356deSJérôme Duval 32857e2f323SJérôme Duval buttonRect = BRect(BPoint(0,0), kSpeakerIconBitmapSize); 32957e2f323SJérôme Duval buttonRect.OffsetTo(background->Bounds().RightBottom() - BPoint(121, 17)); 330ad0c65eaSJérôme Duval SpeakerView *speakerView = new SpeakerView(buttonRect, 331ad0c65eaSJérôme Duval B_FOLLOW_LEFT | B_FOLLOW_TOP); 33257e2f323SJérôme Duval speakerView->SetResizingMode(B_FOLLOW_RIGHT | B_FOLLOW_TOP); 33357e2f323SJérôme Duval background->AddChild(speakerView); 33457e2f323SJérôme Duval 33557e2f323SJérôme Duval buttonRect = BRect(BPoint(0,0), BPoint(84, 19)); 33657e2f323SJérôme Duval buttonRect.OffsetTo(background->Bounds().RightBottom() - BPoint(107, 20)); 337ad0c65eaSJérôme Duval fVolumeSlider = new VolumeSlider(buttonRect, "volumeSlider", 338ad0c65eaSJérôme Duval B_FOLLOW_LEFT | B_FOLLOW_TOP); 33957e2f323SJérôme Duval fVolumeSlider->SetResizingMode(B_FOLLOW_RIGHT | B_FOLLOW_TOP); 34057e2f323SJérôme Duval background->AddChild(fVolumeSlider); 34157e2f323SJérôme Duval 34257e2f323SJérôme Duval // Button to mask/see sounds list 34357e2f323SJérôme Duval buttonRect = BRect(BPoint(0,0), kUpDownButtonSize); 34457e2f323SJérôme Duval buttonRect.OffsetTo(background->Bounds().RightBottom() - BPoint(21, 25)); 34557e2f323SJérôme Duval fUpDownButton = new UpDownButton(buttonRect, new BMessage(VIEW_LIST)); 34657e2f323SJérôme Duval fUpDownButton->SetResizingMode(B_FOLLOW_RIGHT | B_FOLLOW_TOP); 34757e2f323SJérôme Duval background->AddChild(fUpDownButton); 34857e2f323SJérôme Duval 34957e2f323SJérôme Duval r = Bounds(); 35057e2f323SJérôme Duval r.top = background->Bounds().bottom + 1; 35157e2f323SJérôme Duval fBottomBox = new BBox(r, "bottomBox", B_FOLLOW_ALL); 35257e2f323SJérôme Duval fBottomBox->SetBorder(B_NO_BORDER); 35357e2f323SJérôme Duval AddChild(fBottomBox); 35457e2f323SJérôme Duval 35557e2f323SJérôme Duval // The actual list of recorded sounds (initially empty) sits 35657e2f323SJérôme Duval // below the header with the controls. 35757e2f323SJérôme Duval r = fBottomBox->Bounds(); 35857e2f323SJérôme Duval r.left += 190; 35957e2f323SJérôme Duval r.InsetBy(10, 10); 36057e2f323SJérôme Duval r.left -= 10; 36157e2f323SJérôme Duval r.top += 4; 36257e2f323SJérôme Duval r.right -= B_V_SCROLL_BAR_WIDTH; 36357e2f323SJérôme Duval r.bottom -= 25; 3643ee96407SJérôme Duval fSoundList = new SoundListView(r, B_TRANSLATE("Sound List"), 3653ee96407SJérôme Duval B_FOLLOW_ALL); 36657e2f323SJérôme Duval fSoundList->SetSelectionMessage(new BMessage(SOUND_SELECTED)); 367b271ff3fSPhilippe Saint-Pierre fSoundList->SetViewColor(ui_color(B_PANEL_BACKGROUND_COLOR)); 368ad0c65eaSJérôme Duval BScrollView *scroller = new BScrollView("scroller", fSoundList, 369ad0c65eaSJérôme Duval B_FOLLOW_ALL, 0, false, true, B_FANCY_BORDER); 37057e2f323SJérôme Duval fBottomBox->AddChild(scroller); 37157e2f323SJérôme Duval 37257e2f323SJérôme Duval r = fBottomBox->Bounds(); 37357e2f323SJérôme Duval r.right = r.left + 190; 37457e2f323SJérôme Duval r.bottom -= 25; 37557e2f323SJérôme Duval r.InsetBy(10, 8); 37657e2f323SJérôme Duval r.top -= 1; 377b271ff3fSPhilippe Saint-Pierre fFileInfoBox = new BBox(r, "fileinfo", B_FOLLOW_LEFT); 3783ee96407SJérôme Duval fFileInfoBox->SetLabel(B_TRANSLATE("File info")); 37957e2f323SJérôme Duval 380b271ff3fSPhilippe Saint-Pierre fFileInfoBox->SetHighColor(ui_color(B_PANEL_TEXT_COLOR)); 381b271ff3fSPhilippe Saint-Pierre 382b271ff3fSPhilippe Saint-Pierre BFont font = be_plain_font; 383b271ff3fSPhilippe Saint-Pierre font.SetSize(font.Size() * 0.92f); 384b271ff3fSPhilippe Saint-Pierre font_height height; 385b271ff3fSPhilippe Saint-Pierre font.GetHeight(&height); 386b271ff3fSPhilippe Saint-Pierre float fontHeight = height.ascent + height.leading + height.descent; 387b271ff3fSPhilippe Saint-Pierre 38857e2f323SJérôme Duval r = fFileInfoBox->Bounds(); 38957e2f323SJérôme Duval r.left = 8; 390b271ff3fSPhilippe Saint-Pierre r.top = fontHeight + 6; 391b271ff3fSPhilippe Saint-Pierre r.bottom = r.top + fontHeight + 3; 39257e2f323SJérôme Duval r.right -= 10; 3933ee96407SJérôme Duval fFilename = new BStringView(r, "filename", B_TRANSLATE("File name:")); 39457e2f323SJérôme Duval fFileInfoBox->AddChild(fFilename); 395b271ff3fSPhilippe Saint-Pierre fFilename->SetFont(&font, B_FONT_SIZE); 396b271ff3fSPhilippe Saint-Pierre fFilename->SetHighColor(ui_color(B_PANEL_TEXT_COLOR)); 397b271ff3fSPhilippe Saint-Pierre 398b271ff3fSPhilippe Saint-Pierre r.top += fontHeight; 399b271ff3fSPhilippe Saint-Pierre r.bottom = r.top + fontHeight + 3; 4003ee96407SJérôme Duval fFormat = new BStringView(r, "format", B_TRANSLATE("Format:")); 40157e2f323SJérôme Duval fFileInfoBox->AddChild(fFormat); 402b271ff3fSPhilippe Saint-Pierre fFormat->SetFont(&font, B_FONT_SIZE); 403b271ff3fSPhilippe Saint-Pierre fFormat->SetHighColor(ui_color(B_PANEL_TEXT_COLOR)); 404b271ff3fSPhilippe Saint-Pierre 405b271ff3fSPhilippe Saint-Pierre r.top += fontHeight; 406b271ff3fSPhilippe Saint-Pierre r.bottom = r.top + fontHeight + 3; 4073ee96407SJérôme Duval fCompression = new BStringView(r, "compression", 4083ee96407SJérôme Duval B_TRANSLATE("Compression:")); 40957e2f323SJérôme Duval fFileInfoBox->AddChild(fCompression); 410b271ff3fSPhilippe Saint-Pierre fCompression->SetFont(&font, B_FONT_SIZE); 411b271ff3fSPhilippe Saint-Pierre fCompression->SetHighColor(ui_color(B_PANEL_TEXT_COLOR)); 412b271ff3fSPhilippe Saint-Pierre 413b271ff3fSPhilippe Saint-Pierre r.top += fontHeight; 414b271ff3fSPhilippe Saint-Pierre r.bottom = r.top + fontHeight + 3; 4153ee96407SJérôme Duval fChannels = new BStringView(r, "channels", B_TRANSLATE("Channels:")); 41657e2f323SJérôme Duval fFileInfoBox->AddChild(fChannels); 417b271ff3fSPhilippe Saint-Pierre fChannels->SetFont(&font, B_FONT_SIZE); 418b271ff3fSPhilippe Saint-Pierre fChannels->SetHighColor(ui_color(B_PANEL_TEXT_COLOR)); 419b271ff3fSPhilippe Saint-Pierre 420b271ff3fSPhilippe Saint-Pierre r.top += fontHeight; 421b271ff3fSPhilippe Saint-Pierre r.bottom = r.top + fontHeight + 3; 4223ee96407SJérôme Duval fSampleSize = new BStringView(r, "samplesize", 4233ee96407SJérôme Duval B_TRANSLATE("Sample size:")); 42457e2f323SJérôme Duval fFileInfoBox->AddChild(fSampleSize); 425b271ff3fSPhilippe Saint-Pierre fSampleSize->SetFont(&font, B_FONT_SIZE); 426b271ff3fSPhilippe Saint-Pierre fSampleSize->SetHighColor(ui_color(B_PANEL_TEXT_COLOR)); 427b271ff3fSPhilippe Saint-Pierre 428b271ff3fSPhilippe Saint-Pierre r.top += fontHeight; 429b271ff3fSPhilippe Saint-Pierre r.bottom = r.top + fontHeight + 3; 4303ee96407SJérôme Duval fSampleRate = new BStringView(r, "samplerate", 4313ee96407SJérôme Duval B_TRANSLATE("Sample rate:")); 43257e2f323SJérôme Duval fFileInfoBox->AddChild(fSampleRate); 433b271ff3fSPhilippe Saint-Pierre fSampleRate->SetFont(&font, B_FONT_SIZE); 434b271ff3fSPhilippe Saint-Pierre fSampleRate->SetHighColor(ui_color(B_PANEL_TEXT_COLOR)); 435b271ff3fSPhilippe Saint-Pierre 436b271ff3fSPhilippe Saint-Pierre r.top += fontHeight; 437b271ff3fSPhilippe Saint-Pierre r.bottom = r.top + fontHeight + 3; 4383ee96407SJérôme Duval fDuration = new BStringView(r, "duration", B_TRANSLATE("Duration:")); 43957e2f323SJérôme Duval fFileInfoBox->AddChild(fDuration); 440b271ff3fSPhilippe Saint-Pierre fDuration->SetFont(&font, B_FONT_SIZE); 441b271ff3fSPhilippe Saint-Pierre fDuration->SetHighColor(ui_color(B_PANEL_TEXT_COLOR)); 442b271ff3fSPhilippe Saint-Pierre 443b271ff3fSPhilippe Saint-Pierre fFileInfoBox->ResizeTo(fFileInfoBox->Frame().Width(), 444b271ff3fSPhilippe Saint-Pierre r.bottom + fontHeight / 2.0f); 445b271ff3fSPhilippe Saint-Pierre fDeployedHeight = MIN_HEIGHT + fFileInfoBox->Bounds().Height() + 40.0f; 44657e2f323SJérôme Duval 44757e2f323SJérôme Duval // Input selection lists all available physical inputs that produce 44857e2f323SJérôme Duval // buffers with B_MEDIA_RAW_AUDIO format data. 4493ee96407SJérôme Duval popup = new BPopUpMenu(B_TRANSLATE("Input")); 450a8ee8b8fSStephan Aßmus const int maxInputCount = 64; 451a8ee8b8fSStephan Aßmus dormant_node_info dni[maxInputCount]; 45257e2f323SJérôme Duval 453a8ee8b8fSStephan Aßmus int32 real_count = maxInputCount; 45457e2f323SJérôme Duval media_format output_format; 45557e2f323SJérôme Duval output_format.type = B_MEDIA_RAW_AUDIO; 45657e2f323SJérôme Duval output_format.u.raw_audio = media_raw_audio_format::wildcard; 45757e2f323SJérôme Duval error = fRoster->GetDormantNodes(dni, &real_count, 0, &output_format, 45857e2f323SJérôme Duval 0, B_BUFFER_PRODUCER | B_PHYSICAL_INPUT); 459a8ee8b8fSStephan Aßmus if (real_count > maxInputCount) { 460a8ee8b8fSStephan Aßmus WINDOW((stderr, "dropped %ld inputs\n", real_count - maxInputCount)); 461a8ee8b8fSStephan Aßmus real_count = maxInputCount; 46257e2f323SJérôme Duval } 4637974d3dcSStephan Aßmus char selected_name[B_MEDIA_NAME_LENGTH] = "Default input"; 46457e2f323SJérôme Duval BMessage * msg; 46557e2f323SJérôme Duval BMenuItem * item; 466ad0c65eaSJérôme Duval for (int i = 0; i < real_count; i++) { 46757e2f323SJérôme Duval msg = new BMessage(INPUT_SELECTED); 468ad0c65eaSJérôme Duval msg->AddData("node", B_RAW_TYPE, &dni[i], sizeof(dni[i])); 469ad0c65eaSJérôme Duval item = new BMenuItem(dni[i].name, msg); 47057e2f323SJérôme Duval popup->AddItem(item); 47157e2f323SJérôme Duval media_node_id ni[12]; 47257e2f323SJérôme Duval int32 ni_count = 12; 473ad0c65eaSJérôme Duval error = fRoster->GetInstancesFor(dni[i].addon, dni[i].flavor_id, 474ad0c65eaSJérôme Duval ni, &ni_count); 475b1ed9a63SJérôme Duval if (error == B_OK) { 476b1ed9a63SJérôme Duval for (int j = 0; j < ni_count; j++) { 477ad0c65eaSJérôme Duval if (ni[j] == fAudioInputNode.node) { 478ad0c65eaSJérôme Duval strcpy(selected_name, dni[i].name); 47957e2f323SJérôme Duval break; 48057e2f323SJérôme Duval } 48157e2f323SJérôme Duval } 482b1ed9a63SJérôme Duval } 483b1ed9a63SJérôme Duval } 48457e2f323SJérôme Duval 48557e2f323SJérôme Duval // Create the actual widget 486b271ff3fSPhilippe Saint-Pierre r = fFileInfoBox->Bounds(); 487b271ff3fSPhilippe Saint-Pierre r.top = r.bottom + 2; 488b271ff3fSPhilippe Saint-Pierre r.bottom = r.top + 18; 48957e2f323SJérôme Duval r.InsetBy(10, 10); 4903ee96407SJérôme Duval fInputField = new BMenuField(r, "Input", B_TRANSLATE("Input:"), popup); 491b1ed9a63SJérôme Duval fInputField->SetDivider(fInputField->StringWidth(B_TRANSLATE("Input:")) 492b1ed9a63SJérôme Duval + 4.0f); 49357e2f323SJérôme Duval fBottomBox->AddChild(fInputField); 49457e2f323SJérôme Duval 49557e2f323SJérôme Duval fBottomBox->AddChild(fFileInfoBox); 49657e2f323SJérôme Duval 49757e2f323SJérôme Duval fBottomBox->Hide(); 498b271ff3fSPhilippe Saint-Pierre CalcSizes(MIN_WIDTH, MIN_HEIGHT); 499b271ff3fSPhilippe Saint-Pierre ResizeTo(Frame().Width(), MIN_HEIGHT); 50057e2f323SJérôme Duval 50157e2f323SJérôme Duval popup->Superitem()->SetLabel(selected_name); 50257e2f323SJérôme Duval 50357e2f323SJérôme Duval // Make sure the save panel is happy. 5045fa77532SJérôme Duval fSavePanel = new BFilePanel(B_SAVE_PANEL); 5055fa77532SJérôme Duval fSavePanel->SetTarget(this); 50657e2f323SJérôme Duval } 50757e2f323SJérôme Duval catch (...) { 50857e2f323SJérôme Duval goto bad_mojo; 50957e2f323SJérôme Duval } 51057e2f323SJérôme Duval UpdateButtons(); 51157e2f323SJérôme Duval return B_OK; 51257e2f323SJérôme Duval 51357e2f323SJérôme Duval // Error handling. 51457e2f323SJérôme Duval bad_mojo: 515b1ed9a63SJérôme Duval if (error >= 0) 51657e2f323SJérôme Duval error = B_ERROR; 517b1ed9a63SJérôme Duval if (fRecordNode) 51857e2f323SJérôme Duval fRecordNode->Release(); 51957e2f323SJérôme Duval 52057e2f323SJérôme Duval delete fPlayer; 521b1ed9a63SJérôme Duval if (!fInputField) 52257e2f323SJérôme Duval delete popup; 52357e2f323SJérôme Duval return error; 52457e2f323SJérôme Duval } 52557e2f323SJérôme Duval 52657e2f323SJérôme Duval 52757e2f323SJérôme Duval bool 52857e2f323SJérôme Duval RecorderWindow::QuitRequested() // this means Close pressed 52957e2f323SJérôme Duval { 53057e2f323SJérôme Duval StopRecording(); 53157e2f323SJérôme Duval StopPlaying(); 53257e2f323SJérôme Duval be_app->PostMessage(B_QUIT_REQUESTED); 53357e2f323SJérôme Duval return true; 53457e2f323SJérôme Duval } 53557e2f323SJérôme Duval 53657e2f323SJérôme Duval 53757e2f323SJérôme Duval void 53857e2f323SJérôme Duval RecorderWindow::MessageReceived(BMessage * message) 53957e2f323SJérôme Duval { 54057e2f323SJérôme Duval // Your average generic message dispatching switch() statement. 54157e2f323SJérôme Duval switch (message->what) { 54257e2f323SJérôme Duval case INPUT_SELECTED: 54357e2f323SJérôme Duval Input(message); 54457e2f323SJérôme Duval break; 54557e2f323SJérôme Duval case SOUND_SELECTED: 54657e2f323SJérôme Duval Selected(message); 54757e2f323SJérôme Duval break; 54857e2f323SJérôme Duval case STOP_PLAYING: 54957e2f323SJérôme Duval StopPlaying(); 55057e2f323SJérôme Duval break; 55157e2f323SJérôme Duval case STOP_RECORDING: 55257e2f323SJérôme Duval StopRecording(); 55357e2f323SJérôme Duval break; 55457e2f323SJérôme Duval case PLAY_PERIOD: 55557e2f323SJérôme Duval if (fPlayer) { 55657e2f323SJérôme Duval if (fPlayer->HasData()) 55757e2f323SJérôme Duval fPlayButton->SetPlaying(); 55857e2f323SJérôme Duval else 55957e2f323SJérôme Duval fPlayButton->SetPaused(); 56057e2f323SJérôme Duval } 56157e2f323SJérôme Duval break; 56257e2f323SJérôme Duval case RECORD_PERIOD: 56357e2f323SJérôme Duval fRecordButton->SetRecording(); 56457e2f323SJérôme Duval break; 56557e2f323SJérôme Duval case RECORD: 56657e2f323SJérôme Duval Record(message); 56757e2f323SJérôme Duval break; 56857e2f323SJérôme Duval case STOP: 56957e2f323SJérôme Duval Stop(message); 57057e2f323SJérôme Duval break; 57157e2f323SJérôme Duval case PLAY: 57257e2f323SJérôme Duval Play(message); 57357e2f323SJérôme Duval break; 57457e2f323SJérôme Duval case SAVE: 57557e2f323SJérôme Duval Save(message); 57657e2f323SJérôme Duval break; 57757e2f323SJérôme Duval case B_SAVE_REQUESTED: 57857e2f323SJérôme Duval DoSave(message); 57957e2f323SJérôme Duval break; 58057e2f323SJérôme Duval case VIEW_LIST: 58157e2f323SJérôme Duval if (fUpDownButton->Value() == B_CONTROL_ON) { 58257e2f323SJérôme Duval fBottomBox->Show(); 583b271ff3fSPhilippe Saint-Pierre CalcSizes(MIN_WIDTH, fDeployedHeight); 584b271ff3fSPhilippe Saint-Pierre ResizeTo(Frame().Width(), fDeployedHeight); 58557e2f323SJérôme Duval } else { 58657e2f323SJérôme Duval fBottomBox->Hide(); 587b271ff3fSPhilippe Saint-Pierre CalcSizes(MIN_WIDTH, MIN_HEIGHT); 588b271ff3fSPhilippe Saint-Pierre ResizeTo(Frame().Width(), MIN_HEIGHT); 58957e2f323SJérôme Duval 59057e2f323SJérôme Duval } 59157e2f323SJérôme Duval break; 59257e2f323SJérôme Duval case UPDATE_TRACKSLIDER: 59357e2f323SJérôme Duval { 59457e2f323SJérôme Duval bigtime_t timestamp = fPlayTrack->CurrentTime(); 59557e2f323SJérôme Duval fTrackSlider->SetMainTime(timestamp, false); 59657e2f323SJérôme Duval fScopeView->SetMainTime(timestamp); 59757e2f323SJérôme Duval } 59857e2f323SJérôme Duval break; 59957e2f323SJérôme Duval case POSITION_CHANGED: 60057e2f323SJérôme Duval { 60157e2f323SJérôme Duval bigtime_t right, left, main; 60257e2f323SJérôme Duval if (message->FindInt64("main", &main) == B_OK) { 60357e2f323SJérôme Duval if (fPlayTrack) { 60457e2f323SJérôme Duval fPlayTrack->SeekToTime(fTrackSlider->MainTime()); 60557e2f323SJérôme Duval fPlayFrame = fPlayTrack->CurrentFrame(); 60657e2f323SJérôme Duval } 60757e2f323SJérôme Duval fScopeView->SetMainTime(main); 60857e2f323SJérôme Duval } 60957e2f323SJérôme Duval if (message->FindInt64("right", &right) == B_OK) { 610b1ed9a63SJérôme Duval if (fPlayTrack) { 611ad0c65eaSJérôme Duval fPlayLimit = MIN(fPlayFrames, 612b1ed9a63SJérôme Duval (off_t)(right * fPlayFormat.u.raw_audio.frame_rate 613b1ed9a63SJérôme Duval / 1000000LL)); 614b1ed9a63SJérôme Duval } 61557e2f323SJérôme Duval fScopeView->SetRightTime(right); 61657e2f323SJérôme Duval } 617b1ed9a63SJérôme Duval if (message->FindInt64("left", &left) == B_OK) 61857e2f323SJérôme Duval fScopeView->SetLeftTime(left); 61957e2f323SJérôme Duval break; 620b1ed9a63SJérôme Duval } 621948356deSJérôme Duval case LOOP: 622948356deSJérôme Duval fLooping = fLoopButton->ButtonState(); 623948356deSJérôme Duval break; 62457e2f323SJérôme Duval case B_SIMPLE_DATA: 62557e2f323SJérôme Duval case B_REFS_RECEIVED: 62657e2f323SJérôme Duval { 62757e2f323SJérôme Duval RefsReceived(message); 62857e2f323SJérôme Duval break; 62957e2f323SJérôme Duval } 630019ed09bSJérôme Duval case B_COPY_TARGET: 631019ed09bSJérôme Duval CopyTarget(message); 632019ed09bSJérôme Duval break; 63357e2f323SJérôme Duval default: 63457e2f323SJérôme Duval BWindow::MessageReceived(message); 63557e2f323SJérôme Duval break; 63657e2f323SJérôme Duval } 63757e2f323SJérôme Duval } 63857e2f323SJérôme Duval 63957e2f323SJérôme Duval 64057e2f323SJérôme Duval void 64157e2f323SJérôme Duval RecorderWindow::Record(BMessage * message) 64257e2f323SJérôme Duval { 64357e2f323SJérôme Duval // User pressed Record button 64457e2f323SJérôme Duval fRecording = true; 64557e2f323SJérôme Duval if (fButtonState != btnPaused) { 64657e2f323SJérôme Duval StopRecording(); 64757e2f323SJérôme Duval return; // user is too fast on the mouse 64857e2f323SJérôme Duval } 64957e2f323SJérôme Duval SetButtonState(btnRecording); 65057e2f323SJérôme Duval fRecordButton->SetRecording(); 65157e2f323SJérôme Duval 65257e2f323SJérôme Duval char name[256]; 65357e2f323SJérôme Duval // Create a file with a temporary name 65457e2f323SJérôme Duval status_t err = NewTempName(name); 65557e2f323SJérôme Duval if (err < B_OK) { 6563ee96407SJérôme Duval ErrorAlert(B_TRANSLATE("Cannot find an unused name to use for the " 6573ee96407SJérôme Duval "new recording"), err); 65857e2f323SJérôme Duval return; 65957e2f323SJérôme Duval } 66057e2f323SJérôme Duval // Find the file so we can refer to it later 66157e2f323SJérôme Duval err = fTempDir.FindEntry(name, &fRecEntry); 66257e2f323SJérôme Duval if (err < B_OK) { 6633ee96407SJérôme Duval ErrorAlert(B_TRANSLATE("Cannot find the temporary file created to " 6643ee96407SJérôme Duval "hold the new recording"), err); 66557e2f323SJérôme Duval return; 66657e2f323SJérôme Duval } 66757e2f323SJérôme Duval err = fRecFile.SetTo(&fTempDir, name, O_RDWR); 66857e2f323SJérôme Duval if (err < B_OK) { 6693ee96407SJérôme Duval ErrorAlert(B_TRANSLATE("Cannot open the temporary file created to " 6703ee96407SJérôme Duval "hold the new recording"), err); 67157e2f323SJérôme Duval fRecEntry.Unset(); 67257e2f323SJérôme Duval return; 67357e2f323SJérôme Duval } 67457e2f323SJérôme Duval // Reserve space on disk (creates fewer fragments) 67507db0c6fSJérôme Duval err = fRecFile.SetSize(4 * fRecordFormat.u.raw_audio.channel_count 6763ee96407SJérôme Duval * fRecordFormat.u.raw_audio.frame_rate 6773ee96407SJérôme Duval * (fRecordFormat.u.raw_audio.format 678ad0c65eaSJérôme Duval & media_raw_audio_format::B_AUDIO_SIZE_MASK)); 67957e2f323SJérôme Duval if (err < B_OK) { 6803ee96407SJérôme Duval ErrorAlert(B_TRANSLATE("Cannot record a sound that long"), err); 68157e2f323SJérôme Duval fRecEntry.Remove(); 68257e2f323SJérôme Duval fRecEntry.Unset(); 68357e2f323SJérôme Duval return; 68457e2f323SJérôme Duval } 68557e2f323SJérôme Duval fRecSize = 0; 68657e2f323SJérôme Duval 68757e2f323SJérôme Duval fRecFile.Seek(sizeof(struct wave_struct), SEEK_SET); 68857e2f323SJérôme Duval 68957e2f323SJérôme Duval // Hook up input 69057e2f323SJérôme Duval err = MakeRecordConnection(fAudioInputNode); 69157e2f323SJérôme Duval if (err < B_OK) { 6923ee96407SJérôme Duval ErrorAlert(B_TRANSLATE("Cannot connect to the selected sound input"), 6933ee96407SJérôme Duval err); 69457e2f323SJérôme Duval fRecEntry.Remove(); 69557e2f323SJérôme Duval fRecEntry.Unset(); 69657e2f323SJérôme Duval return; 69757e2f323SJérôme Duval } 69857e2f323SJérôme Duval 69957e2f323SJérôme Duval // And get it going... 70057e2f323SJérôme Duval bigtime_t then = fRecordNode->TimeSource()->Now() + 50000LL; 70157e2f323SJérôme Duval fRoster->StartNode(fRecordNode->Node(), then); 70257e2f323SJérôme Duval if (fAudioInputNode.kind & B_TIME_SOURCE) { 703ad0c65eaSJérôme Duval fRoster->StartNode(fAudioInputNode, 704ad0c65eaSJérôme Duval fRecordNode->TimeSource()->RealTimeFor(then, 0)); 705b1ed9a63SJérôme Duval } else 70657e2f323SJérôme Duval fRoster->StartNode(fAudioInputNode, then); 70757e2f323SJérôme Duval } 70857e2f323SJérôme Duval 7098e32a27eSPhilippe Saint-Pierre 71057e2f323SJérôme Duval void 71157e2f323SJérôme Duval RecorderWindow::Play(BMessage * message) 71257e2f323SJérôme Duval { 71357e2f323SJérôme Duval if (fPlayer) { 71457e2f323SJérôme Duval // User pressed Play button and playing 71557e2f323SJérôme Duval if (fPlayer->HasData()) 71657e2f323SJérôme Duval fPlayButton->SetPaused(); 71757e2f323SJérôme Duval else 71857e2f323SJérôme Duval fPlayButton->SetPlaying(); 71957e2f323SJérôme Duval fPlayer->SetHasData(!fPlayer->HasData()); 72057e2f323SJérôme Duval return; 72157e2f323SJérôme Duval } 72257e2f323SJérôme Duval 72357e2f323SJérôme Duval SetButtonState(btnPlaying); 72457e2f323SJérôme Duval fPlayButton->SetPlaying(); 72557e2f323SJérôme Duval 72657e2f323SJérôme Duval if (!fPlayTrack) { 7273ee96407SJérôme Duval ErrorAlert(B_TRANSLATE("Cannot get the file to play"), B_ERROR); 72857e2f323SJérôme Duval return; 72957e2f323SJérôme Duval } 73057e2f323SJérôme Duval 731ad0c65eaSJérôme Duval fPlayLimit = MIN(fPlayFrames, (off_t)(fTrackSlider->RightTime() 732ad0c65eaSJérôme Duval * fPlayFormat.u.raw_audio.frame_rate / 1000000LL)); 73357e2f323SJérôme Duval fPlayTrack->SeekToTime(fTrackSlider->MainTime()); 73457e2f323SJérôme Duval fPlayFrame = fPlayTrack->CurrentFrame(); 73557e2f323SJérôme Duval 73657e2f323SJérôme Duval // Create our internal Node which plays sound, and register it. 7373ee96407SJérôme Duval fPlayer = new BSoundPlayer(fAudioMixerNode, &fPlayFormat.u.raw_audio, 7383ee96407SJérôme Duval "Sound Player"); 73957e2f323SJérôme Duval status_t err = fPlayer->InitCheck(); 740b1ed9a63SJérôme Duval if (err < B_OK) 74157e2f323SJérôme Duval return; 74257e2f323SJérôme Duval 74357e2f323SJérôme Duval fVolumeSlider->SetSoundPlayer(fPlayer); 74457e2f323SJérôme Duval fPlayer->SetCallbacks(PlayFile, NotifyPlayFile, this); 74557e2f323SJérôme Duval 74657e2f323SJérôme Duval // And get it going... 74757e2f323SJérôme Duval fPlayer->Start(); 74857e2f323SJérôme Duval fPlayer->SetHasData(true); 74957e2f323SJérôme Duval } 75057e2f323SJérôme Duval 7518e32a27eSPhilippe Saint-Pierre 75257e2f323SJérôme Duval void 75357e2f323SJérôme Duval RecorderWindow::Stop(BMessage * message) 75457e2f323SJérôme Duval { 75557e2f323SJérôme Duval // User pressed Stop button. 75657e2f323SJérôme Duval // Stop recorder. 75757e2f323SJérôme Duval StopRecording(); 75857e2f323SJérôme Duval // Stop player. 75957e2f323SJérôme Duval StopPlaying(); 76057e2f323SJérôme Duval } 76157e2f323SJérôme Duval 7628e32a27eSPhilippe Saint-Pierre 76357e2f323SJérôme Duval void 76457e2f323SJérôme Duval RecorderWindow::Save(BMessage * message) 76557e2f323SJérôme Duval { 76657e2f323SJérôme Duval // User pressed Save button. 76757e2f323SJérôme Duval // Find the item to save. 76857e2f323SJérôme Duval int32 index = fSoundList->CurrentSelection(); 76957e2f323SJérôme Duval SoundListItem* pItem = dynamic_cast<SoundListItem*>(fSoundList->ItemAt(index)); 770b1ed9a63SJérôme Duval if ((! pItem) || (pItem->Entry().InitCheck() != B_OK)) 77157e2f323SJérôme Duval return; 77257e2f323SJérôme Duval 77357e2f323SJérôme Duval // Update the save panel and show it. 77457e2f323SJérôme Duval char filename[B_FILE_NAME_LENGTH]; 77557e2f323SJérôme Duval pItem->Entry().GetName(filename); 77657e2f323SJérôme Duval BMessage saveMsg(B_SAVE_REQUESTED); 77757e2f323SJérôme Duval entry_ref ref; 77857e2f323SJérôme Duval pItem->Entry().GetRef(&ref); 77957e2f323SJérôme Duval 780a4797804SJérôme Duval if (saveMsg.AddPointer("sound list item", pItem) != B_OK) 781a4797804SJérôme Duval fprintf(stderr, "failed to add pItem\n"); 7825fa77532SJérôme Duval fSavePanel->SetSaveText(filename); 7835fa77532SJérôme Duval fSavePanel->SetMessage(&saveMsg); 7845fa77532SJérôme Duval fSavePanel->Show(); 78557e2f323SJérôme Duval } 78657e2f323SJérôme Duval 7878e32a27eSPhilippe Saint-Pierre 78857e2f323SJérôme Duval void 78957e2f323SJérôme Duval RecorderWindow::DoSave(BMessage * message) 79057e2f323SJérôme Duval { 79157e2f323SJérôme Duval // User picked a place to put the file. 79257e2f323SJérôme Duval // Find the location of the old (e.g. 79357e2f323SJérôme Duval // temporary file), and the name of the 79457e2f323SJérôme Duval // new file to save. 79557e2f323SJérôme Duval entry_ref old_ref, new_dir_ref; 79657e2f323SJérôme Duval const char* new_name; 79757e2f323SJérôme Duval SoundListItem* pItem; 79857e2f323SJérôme Duval 79957e2f323SJérôme Duval if ((message->FindPointer("sound list item", (void**) &pItem) == B_OK) 80057e2f323SJérôme Duval && (message->FindRef("directory", &new_dir_ref) == B_OK) 801b1ed9a63SJérôme Duval && (message->FindString("name", &new_name) == B_OK)) { 80257e2f323SJérôme Duval BEntry& oldEntry = pItem->Entry(); 80357e2f323SJérôme Duval BFile oldFile(&oldEntry, B_READ_WRITE); 80457e2f323SJérôme Duval if (oldFile.InitCheck() != B_OK) 80557e2f323SJérôme Duval return; 80657e2f323SJérôme Duval 80757e2f323SJérôme Duval BDirectory newDir(&new_dir_ref); 80857e2f323SJérôme Duval if (newDir.InitCheck() != B_OK) 80957e2f323SJérôme Duval return; 81057e2f323SJérôme Duval 81157e2f323SJérôme Duval BFile newFile; 81257e2f323SJérôme Duval newDir.CreateFile(new_name, &newFile); 81357e2f323SJérôme Duval 81457e2f323SJérôme Duval if (newFile.InitCheck() != B_OK) 81557e2f323SJérôme Duval return; 81657e2f323SJérôme Duval 81757e2f323SJérôme Duval status_t err = CopyFile(newFile, oldFile); 81857e2f323SJérôme Duval 81957e2f323SJérôme Duval if (err == B_OK) { 82057e2f323SJérôme Duval // clean up the sound list and item 82157e2f323SJérôme Duval if (pItem->IsTemp()) 82257e2f323SJérôme Duval oldEntry.Remove(); // blows away temp file! 82357e2f323SJérôme Duval oldEntry.SetTo(&newDir, new_name); 82457e2f323SJérôme Duval pItem->SetTemp(false); // don't blow the new entry away when we exit! 82557e2f323SJérôme Duval fSoundList->Invalidate(); 82657e2f323SJérôme Duval } 82757e2f323SJérôme Duval } else { 82857e2f323SJérôme Duval WINDOW((stderr, "Couldn't save file.\n")); 82957e2f323SJérôme Duval } 83057e2f323SJérôme Duval } 83157e2f323SJérôme Duval 83257e2f323SJérôme Duval 83357e2f323SJérôme Duval void 83457e2f323SJérôme Duval RecorderWindow::Input(BMessage * message) 83557e2f323SJérôme Duval { 83657e2f323SJérôme Duval // User selected input from pop-up 83757e2f323SJérôme Duval const dormant_node_info * dni = 0; 83857e2f323SJérôme Duval ssize_t size = 0; 839b1ed9a63SJérôme Duval if (message->FindData("node", B_RAW_TYPE, (const void **)&dni, &size)) 84057e2f323SJérôme Duval return; // bad input selection message 84157e2f323SJérôme Duval 84257e2f323SJérôme Duval media_node_id node_id; 84357e2f323SJérôme Duval status_t error = fRoster->GetInstancesFor(dni->addon, dni->flavor_id, &node_id); 844b1ed9a63SJérôme Duval if (error != B_OK) 84557e2f323SJérôme Duval fRoster->InstantiateDormantNode(*dni, &fAudioInputNode); 846b1ed9a63SJérôme Duval else 84757e2f323SJérôme Duval fRoster->GetNodeFor(node_id, &fAudioInputNode); 84857e2f323SJérôme Duval } 84957e2f323SJérôme Duval 8508e32a27eSPhilippe Saint-Pierre 85157e2f323SJérôme Duval void 85257e2f323SJérôme Duval RecorderWindow::Selected(BMessage * message) 85357e2f323SJérôme Duval { 85457e2f323SJérôme Duval // User selected a sound in list view 855c6f8aa29SJérôme Duval int32 selIdx = fSoundList->CurrentSelection(); 856c6f8aa29SJérôme Duval SoundListItem* pItem = dynamic_cast<SoundListItem*>(fSoundList->ItemAt(selIdx)); 857c6f8aa29SJérôme Duval if (!pItem) 858c6f8aa29SJérôme Duval return; 859c6f8aa29SJérôme Duval status_t err = UpdatePlayFile(pItem, true); 860c6f8aa29SJérôme Duval if (err != B_OK) { 8613ee96407SJérôme Duval ErrorAlert(B_TRANSLATE("Cannot recognize this file as a media file"), 862ad0c65eaSJérôme Duval err == B_MEDIA_NO_HANDLER ? B_OK : err); 863c6f8aa29SJérôme Duval RemoveCurrentSoundItem(); 864c6f8aa29SJérôme Duval } 86557e2f323SJérôme Duval UpdateButtons(); 86657e2f323SJérôme Duval } 86757e2f323SJérôme Duval 8688e32a27eSPhilippe Saint-Pierre 86957e2f323SJérôme Duval status_t 87057e2f323SJérôme Duval RecorderWindow::MakeRecordConnection(const media_node & input) 87157e2f323SJérôme Duval { 87257e2f323SJérôme Duval CONNECT((stderr, "RecorderWindow::MakeRecordConnection()\n")); 87357e2f323SJérôme Duval 87457e2f323SJérôme Duval // Find an available output for the given input node. 87557e2f323SJérôme Duval int32 count = 0; 87657e2f323SJérôme Duval status_t err = fRoster->GetFreeOutputsFor(input, &fAudioOutput, 1, &count, B_MEDIA_RAW_AUDIO); 87757e2f323SJérôme Duval if (err < B_OK) { 878ad0c65eaSJérôme Duval CONNECT((stderr, "RecorderWindow::MakeRecordConnection():" 879ad0c65eaSJérôme Duval " couldn't get free outputs from audio input node\n")); 88057e2f323SJérôme Duval return err; 88157e2f323SJérôme Duval } 88257e2f323SJérôme Duval if (count < 1) { 883ad0c65eaSJérôme Duval CONNECT((stderr, "RecorderWindow::MakeRecordConnection():" 884ad0c65eaSJérôme Duval " no free outputs from audio input node\n")); 88557e2f323SJérôme Duval return B_BUSY; 88657e2f323SJérôme Duval } 88757e2f323SJérôme Duval 88857e2f323SJérôme Duval // Find an available input for our own Node. Note that we go through the 88957e2f323SJérôme Duval // MediaRoster; calling Media Kit methods directly on Nodes in our app is 89057e2f323SJérôme Duval // not OK (because synchronization happens in the service thread, not in 89157e2f323SJérôme Duval // the calling thread). 89257e2f323SJérôme Duval // TODO: explain this 89357e2f323SJérôme Duval err = fRoster->GetFreeInputsFor(fRecordNode->Node(), &fRecInput, 1, &count, B_MEDIA_RAW_AUDIO); 89457e2f323SJérôme Duval if (err < B_OK) { 895ad0c65eaSJérôme Duval CONNECT((stderr, "RecorderWindow::MakeRecordConnection():" 896ad0c65eaSJérôme Duval " couldn't get free inputs for sound recorder\n")); 89757e2f323SJérôme Duval return err; 89857e2f323SJérôme Duval } 89957e2f323SJérôme Duval if (count < 1) { 900ad0c65eaSJérôme Duval CONNECT((stderr, "RecorderWindow::MakeRecordConnection():" 901ad0c65eaSJérôme Duval " no free inputs for sound recorder\n")); 90257e2f323SJérôme Duval return B_BUSY; 90357e2f323SJérôme Duval } 90457e2f323SJérôme Duval 90557e2f323SJérôme Duval // Find out what the time source of the input is. 90657e2f323SJérôme Duval // For most nodes, we just use the preferred time source (the DAC) for synchronization. 90757e2f323SJérôme Duval // However, nodes that record from an input need to synchronize to the audio input node 90857e2f323SJérôme Duval // instead for best results. 90957e2f323SJérôme Duval // MakeTimeSourceFor gives us a "clone" of the time source node that we can manipulate 91057e2f323SJérôme Duval // to our heart's content. When we're done with it, though, we need to call Release() 91157e2f323SJérôme Duval // on the time source node, so that it keeps an accurate reference count and can delete 91257e2f323SJérôme Duval // itself when it's no longer needed. 91357e2f323SJérôme Duval // TODO: what about filters connected to audio input? 91457e2f323SJérôme Duval media_node use_time_source; 91557e2f323SJérôme Duval BTimeSource * tsobj = fRoster->MakeTimeSourceFor(input); 91657e2f323SJérôme Duval if (! tsobj) { 917ad0c65eaSJérôme Duval CONNECT((stderr, "RecorderWindow::MakeRecordConnection():" 918ad0c65eaSJérôme Duval " couldn't clone time source from audio input node\n")); 91957e2f323SJérôme Duval return B_MEDIA_BAD_NODE; 92057e2f323SJérôme Duval } 92157e2f323SJérôme Duval 92257e2f323SJérôme Duval // Apply the time source in effect to our own Node. 92357e2f323SJérôme Duval err = fRoster->SetTimeSourceFor(fRecordNode->Node().node, tsobj->Node().node); 92457e2f323SJérôme Duval if (err < B_OK) { 925ad0c65eaSJérôme Duval CONNECT((stderr, "RecorderWindow::MakeRecordConnection():" 926ad0c65eaSJérôme Duval " couldn't set the sound recorder's time source\n")); 92757e2f323SJérôme Duval tsobj->Release(); 92857e2f323SJérôme Duval return err; 92957e2f323SJérôme Duval } 93057e2f323SJérôme Duval 93157e2f323SJérôme Duval // Get a format, any format. 93288fef303SJérôme Duval fRecordFormat.u.raw_audio = fAudioOutput.format.u.raw_audio; 93388fef303SJérôme Duval fRecordFormat.type = B_MEDIA_RAW_AUDIO; 93457e2f323SJérôme Duval 93557e2f323SJérôme Duval // Tell the consumer where we want data to go. 93657e2f323SJérôme Duval err = fRecordNode->SetHooks(RecordFile, NotifyRecordFile, this); 93757e2f323SJérôme Duval if (err < B_OK) { 938ad0c65eaSJérôme Duval CONNECT((stderr, "RecorderWindow::MakeRecordConnection():" 939ad0c65eaSJérôme Duval " couldn't set the sound recorder's hook functions\n")); 94057e2f323SJérôme Duval tsobj->Release(); 94157e2f323SJérôme Duval return err; 94257e2f323SJérôme Duval } 94357e2f323SJérôme Duval 9443ee96407SJérôme Duval // Using the same structs for input and output is OK in 9453ee96407SJérôme Duval // BMediaRoster::Connect(). 9463ee96407SJérôme Duval err = fRoster->Connect(fAudioOutput.source, fRecInput.destination, 9473ee96407SJérôme Duval &fRecordFormat, &fAudioOutput, &fRecInput); 94857e2f323SJérôme Duval if (err < B_OK) { 949ad0c65eaSJérôme Duval CONNECT((stderr, "RecorderWindow::MakeRecordConnection():" 950ad0c65eaSJérôme Duval " failed to connect sound recorder to audio input node.\n")); 95157e2f323SJérôme Duval tsobj->Release(); 95257e2f323SJérôme Duval fRecordNode->SetHooks(0, 0, 0); 95357e2f323SJérôme Duval return err; 95457e2f323SJérôme Duval } 95557e2f323SJérôme Duval 95657e2f323SJérôme Duval // Start the time source if it's not running. 957b1ed9a63SJérôme Duval if ((tsobj->Node() != input) && !tsobj->IsRunning()) 95857e2f323SJérôme Duval fRoster->StartNode(tsobj->Node(), BTimeSource::RealTime()); 959b1ed9a63SJérôme Duval 96057e2f323SJérôme Duval tsobj->Release(); // we're done with this time source instance! 96157e2f323SJérôme Duval return B_OK; 96257e2f323SJérôme Duval } 96357e2f323SJérôme Duval 96457e2f323SJérôme Duval 96557e2f323SJérôme Duval status_t 96657e2f323SJérôme Duval RecorderWindow::BreakRecordConnection() 96757e2f323SJérôme Duval { 96857e2f323SJérôme Duval status_t err; 96957e2f323SJérôme Duval 97057e2f323SJérôme Duval // If we are the last connection, the Node will stop automatically since it 97157e2f323SJérôme Duval // has nowhere to send data to. 97257e2f323SJérôme Duval err = fRoster->StopNode(fRecInput.node, 0); 973ad0c65eaSJérôme Duval err = fRoster->Disconnect(fAudioOutput.node.node, fAudioOutput.source, 974ad0c65eaSJérôme Duval fRecInput.node.node, fRecInput.destination); 97557e2f323SJérôme Duval fAudioOutput.source = media_source::null; 97657e2f323SJérôme Duval fRecInput.destination = media_destination::null; 97757e2f323SJérôme Duval return err; 97857e2f323SJérôme Duval } 97957e2f323SJérôme Duval 9808e32a27eSPhilippe Saint-Pierre 98157e2f323SJérôme Duval status_t 98257e2f323SJérôme Duval RecorderWindow::StopRecording() 98357e2f323SJérôme Duval { 98457e2f323SJérôme Duval if (!fRecording) 98557e2f323SJérôme Duval return B_OK; 98657e2f323SJérôme Duval fRecording = false; 98757e2f323SJérôme Duval BreakRecordConnection(); 98857e2f323SJérôme Duval fRecordNode->SetHooks(NULL,NULL,NULL); 98957e2f323SJérôme Duval if (fRecSize > 0) { 99057e2f323SJérôme Duval 99157e2f323SJérôme Duval wave_struct header; 99257e2f323SJérôme Duval header.riff.riff_id = FOURCC('R','I','F','F'); 993a4797804SJérôme Duval header.riff.len = fRecSize + sizeof(header) - 8; 99457e2f323SJérôme Duval header.riff.wave_id = FOURCC('W','A','V','E'); 99557e2f323SJérôme Duval header.format_chunk.fourcc = FOURCC('f','m','t',' '); 99657e2f323SJérôme Duval header.format_chunk.len = sizeof(header.format); 99757e2f323SJérôme Duval header.format.format_tag = 1; 99888fef303SJérôme Duval header.format.channels = fRecordFormat.u.raw_audio.channel_count; 99988fef303SJérôme Duval header.format.samples_per_sec = (uint32)fRecordFormat.u.raw_audio.frame_rate; 100088fef303SJérôme Duval header.format.avg_bytes_per_sec = (uint32)(fRecordFormat.u.raw_audio.frame_rate 100188fef303SJérôme Duval * fRecordFormat.u.raw_audio.channel_count 100288fef303SJérôme Duval * (fRecordFormat.u.raw_audio.format & 0xf)); 100388fef303SJérôme Duval header.format.bits_per_sample = (fRecordFormat.u.raw_audio.format & 0xf) * 8; 100488fef303SJérôme Duval header.format.block_align = (fRecordFormat.u.raw_audio.format & 0xf) 100588fef303SJérôme Duval * fRecordFormat.u.raw_audio.channel_count; 100657e2f323SJérôme Duval header.data_chunk.fourcc = FOURCC('d','a','t','a'); 100757e2f323SJérôme Duval header.data_chunk.len = fRecSize; 100857e2f323SJérôme Duval fRecFile.Seek(0, SEEK_SET); 100957e2f323SJérôme Duval fRecFile.Write(&header, sizeof(header)); 101057e2f323SJérôme Duval 10113ee96407SJérôme Duval fRecFile.SetSize(fRecSize + sizeof(header)); 10123ee96407SJérôme Duval // We reserve space; make sure we cut off any excess at the end. 101357e2f323SJérôme Duval AddSoundItem(fRecEntry, true); 1014b1ed9a63SJérôme Duval } else 101557e2f323SJérôme Duval fRecEntry.Remove(); 1016b1ed9a63SJérôme Duval 101757e2f323SJérôme Duval // We're done for this time. 101857e2f323SJérôme Duval fRecEntry.Unset(); 101957e2f323SJérôme Duval // Close the file. 102057e2f323SJérôme Duval fRecFile.Unset(); 102157e2f323SJérôme Duval // No more recording going on. 102257e2f323SJérôme Duval fRecSize = 0; 102357e2f323SJérôme Duval SetButtonState(btnPaused); 102457e2f323SJérôme Duval fRecordButton->SetStopped(); 102557e2f323SJérôme Duval 102657e2f323SJérôme Duval return B_OK; 102757e2f323SJérôme Duval } 102857e2f323SJérôme Duval 102957e2f323SJérôme Duval 103057e2f323SJérôme Duval status_t 103157e2f323SJérôme Duval RecorderWindow::StopPlaying() 103257e2f323SJérôme Duval { 103357e2f323SJérôme Duval if (fPlayer) { 103457e2f323SJérôme Duval fPlayer->Stop(); 103557e2f323SJérôme Duval fPlayer->SetCallbacks(0, 0, 0); 103657e2f323SJérôme Duval fVolumeSlider->SetSoundPlayer(NULL); 103757e2f323SJérôme Duval delete fPlayer; 103857e2f323SJérôme Duval fPlayer = NULL; 103957e2f323SJérôme Duval } 104057e2f323SJérôme Duval SetButtonState(btnPaused); 104157e2f323SJérôme Duval fPlayButton->SetStopped(); 104257e2f323SJérôme Duval fTrackSlider->ResetMainTime(); 104357e2f323SJérôme Duval fScopeView->SetMainTime(*fTrackSlider->MainTime()); 104457e2f323SJérôme Duval return B_OK; 104557e2f323SJérôme Duval } 104657e2f323SJérôme Duval 104757e2f323SJérôme Duval 104857e2f323SJérôme Duval void 104957e2f323SJérôme Duval RecorderWindow::SetButtonState(BtnState state) 105057e2f323SJérôme Duval { 105157e2f323SJérôme Duval fButtonState = state; 105257e2f323SJérôme Duval UpdateButtons(); 105357e2f323SJérôme Duval } 105457e2f323SJérôme Duval 105557e2f323SJérôme Duval 105657e2f323SJérôme Duval void 105757e2f323SJérôme Duval RecorderWindow::UpdateButtons() 105857e2f323SJérôme Duval { 105957e2f323SJérôme Duval bool hasSelection = (fSoundList->CurrentSelection() >= 0); 106057e2f323SJérôme Duval fRecordButton->SetEnabled(fButtonState != btnPlaying); 106157e2f323SJérôme Duval fPlayButton->SetEnabled((fButtonState != btnRecording) && hasSelection); 106257e2f323SJérôme Duval fRewindButton->SetEnabled((fButtonState != btnRecording) && hasSelection); 106357e2f323SJérôme Duval fForwardButton->SetEnabled((fButtonState != btnRecording) && hasSelection); 106457e2f323SJérôme Duval fStopButton->SetEnabled(fButtonState != btnPaused); 106557e2f323SJérôme Duval fSaveButton->SetEnabled(hasSelection && (fButtonState != btnRecording)); 106657e2f323SJérôme Duval fInputField->SetEnabled(fButtonState != btnRecording); 106757e2f323SJérôme Duval } 106857e2f323SJérôme Duval 10695fa77532SJérôme Duval #ifndef __HAIKU__ 10703ee96407SJérôme Duval extern "C" status_t DecodedFormat__11BMediaTrackP12media_format( 10713ee96407SJérôme Duval BMediaTrack *self, media_format *inout_format); 10725fa77532SJérôme Duval #endif 107357e2f323SJérôme Duval 10748e32a27eSPhilippe Saint-Pierre 1075c6f8aa29SJérôme Duval status_t 1076c6f8aa29SJérôme Duval RecorderWindow::UpdatePlayFile(SoundListItem* item, bool updateDisplay) 107757e2f323SJérôme Duval { 1078019ed09bSJérôme Duval fScopeView->CancelRendering(); 1079019ed09bSJérôme Duval StopPlaying(); 1080019ed09bSJérôme Duval StopRecording(); 1081019ed09bSJérôme Duval 1082dbbc34b6SJérôme Duval if (fPlayTrack && fPlayFile) { 10837942339dSJérôme Duval fPlayFile->ReleaseTrack(fPlayTrack); 10847942339dSJérôme Duval fPlayTrack = NULL; 1085dbbc34b6SJérôme Duval } 1086dbbc34b6SJérôme Duval if (fPlayFile) { 1087dbbc34b6SJérôme Duval delete fPlayFile; 10887942339dSJérôme Duval fPlayFile = NULL; 1089dbbc34b6SJérôme Duval } 10907942339dSJérôme Duval 109157e2f323SJérôme Duval status_t err; 1092c6f8aa29SJérôme Duval BEntry& entry = item->Entry(); 109357e2f323SJérôme Duval entry_ref ref; 109457e2f323SJérôme Duval entry.GetRef(&ref); 109557e2f323SJérôme Duval fPlayFile = new BMediaFile(&ref); //, B_MEDIA_FILE_UNBUFFERED); 109657e2f323SJérôme Duval if ((err = fPlayFile->InitCheck()) < B_OK) { 109757e2f323SJérôme Duval delete fPlayFile; 1098dbbc34b6SJérôme Duval fPlayFile = NULL; 1099c6f8aa29SJérôme Duval return err; 110057e2f323SJérôme Duval } 110157e2f323SJérôme Duval 110257e2f323SJérôme Duval for (int ix=0; ix < fPlayFile->CountTracks(); ix++) { 110357e2f323SJérôme Duval BMediaTrack * track = fPlayFile->TrackAt(ix); 110457e2f323SJérôme Duval fPlayFormat.type = B_MEDIA_RAW_AUDIO; 11055fa77532SJérôme Duval #ifdef __HAIKU__ 11065fa77532SJérôme Duval if ((track->DecodedFormat(&fPlayFormat) == B_OK) 11075fa77532SJérôme Duval #else 11085fa77532SJérôme Duval if ((DecodedFormat__11BMediaTrackP12media_format(track, &fPlayFormat) == B_OK) 11095fa77532SJérôme Duval #endif 11105fa77532SJérôme Duval && (fPlayFormat.type == B_MEDIA_RAW_AUDIO)) { 111157e2f323SJérôme Duval fPlayTrack = track; 111257e2f323SJérôme Duval break; 111357e2f323SJérôme Duval } 111457e2f323SJérôme Duval if (track) 111557e2f323SJérôme Duval fPlayFile->ReleaseTrack(track); 111657e2f323SJérôme Duval } 111757e2f323SJérôme Duval 111857e2f323SJérôme Duval if (!fPlayTrack) { 111957e2f323SJérôme Duval delete fPlayFile; 1120019ed09bSJérôme Duval fPlayFile = NULL; 1121c6f8aa29SJérôme Duval return B_STREAM_NOT_FOUND; 112257e2f323SJérôme Duval } 112357e2f323SJérôme Duval 1124c6f8aa29SJérôme Duval if (!updateDisplay) 1125c6f8aa29SJérôme Duval return B_OK; 1126c6f8aa29SJérôme Duval 11273ee96407SJérôme Duval BString filename = B_TRANSLATE("File name: "); 112857e2f323SJérôme Duval filename << ref.name; 112957e2f323SJérôme Duval fFilename->SetText(filename.String()); 113057e2f323SJérôme Duval 11313ee96407SJérôme Duval BString format = B_TRANSLATE("Format: "); 113257e2f323SJérôme Duval media_file_format file_format; 113357e2f323SJérôme Duval if (fPlayFile->GetFileFormatInfo(&file_format) == B_OK) 113457e2f323SJérôme Duval format << file_format.short_name; 11353ee96407SJérôme Duval BString compression = B_TRANSLATE("Compression: "); 113657e2f323SJérôme Duval media_codec_info codec_info; 113757e2f323SJérôme Duval if (fPlayTrack->GetCodecInfo(&codec_info) == B_OK) { 113857e2f323SJérôme Duval if (strcmp(codec_info.short_name, "raw")==0) 11393ee96407SJérôme Duval compression << B_TRANSLATE("None"); 114057e2f323SJérôme Duval else 114157e2f323SJérôme Duval compression << codec_info.short_name; 114257e2f323SJérôme Duval } 11433ee96407SJérôme Duval BString channels = B_TRANSLATE("Channels: "); 114457e2f323SJérôme Duval channels << fPlayFormat.u.raw_audio.channel_count; 11453ee96407SJérôme Duval BString samplesize = B_TRANSLATE("Sample size: "); 11463ee96407SJérôme Duval samplesize << 8 * (fPlayFormat.u.raw_audio.format & 0xf) 11473ee96407SJérôme Duval << B_TRANSLATE(" bits"); 11483ee96407SJérôme Duval BString samplerate = B_TRANSLATE("Sample rate: "); 114957e2f323SJérôme Duval samplerate << (int)fPlayFormat.u.raw_audio.frame_rate; 11503ee96407SJérôme Duval BString durationString = B_TRANSLATE("Duration: "); 115157e2f323SJérôme Duval bigtime_t duration = fPlayTrack->Duration(); 11523ee96407SJérôme Duval durationString << (float)(duration / 1000000.0) << B_TRANSLATE(" seconds"); 115357e2f323SJérôme Duval 115457e2f323SJérôme Duval fFormat->SetText(format.String()); 115557e2f323SJérôme Duval fCompression->SetText(compression.String()); 115657e2f323SJérôme Duval fChannels->SetText(channels.String()); 115757e2f323SJérôme Duval fSampleSize->SetText(samplesize.String()); 115857e2f323SJérôme Duval fSampleRate->SetText(samplerate.String()); 115957e2f323SJérôme Duval fDuration->SetText(durationString.String()); 116057e2f323SJérôme Duval 116157e2f323SJérôme Duval fTrackSlider->SetTotalTime(duration, true); 1162019ed09bSJérôme Duval fScopeView->SetTotalTime(duration, true); 116357e2f323SJérôme Duval fScopeView->RenderTrack(fPlayTrack, fPlayFormat); 11647942339dSJérôme Duval 11657942339dSJérôme Duval fPlayFrames = fPlayTrack->CountFrames(); 1166c6f8aa29SJérôme Duval return B_OK; 116757e2f323SJérôme Duval } 116857e2f323SJérôme Duval 116957e2f323SJérôme Duval 117057e2f323SJérôme Duval void 117157e2f323SJérôme Duval RecorderWindow::ErrorAlert(const char * action, status_t err) 117257e2f323SJérôme Duval { 117357e2f323SJérôme Duval char msg[300]; 1174c6f8aa29SJérôme Duval if (err != B_OK) 11753ee96407SJérôme Duval sprintf(msg, "%s: %s. [%lx]", action, strerror(err), (int32) err); 1176c6f8aa29SJérôme Duval else 11773ee96407SJérôme Duval sprintf(msg, "%s.", action); 1178*aed35104SHumdinger BAlert* alert = new BAlert("", msg, B_TRANSLATE("Stop")); 1179*aed35104SHumdinger alert->SetFlags(alert->Flags() | B_CLOSE_ON_ESCAPE); 1180*aed35104SHumdinger alert->Go(); 118157e2f323SJérôme Duval } 118257e2f323SJérôme Duval 118357e2f323SJérôme Duval 118457e2f323SJérôme Duval status_t 118557e2f323SJérôme Duval RecorderWindow::NewTempName(char * name) 118657e2f323SJérôme Duval { 118757e2f323SJérôme Duval int init_count = fTempCount; 118857e2f323SJérôme Duval again: 118957e2f323SJérôme Duval if (fTempCount-init_count > 25) { 119057e2f323SJérôme Duval return B_ERROR; 119157e2f323SJérôme Duval } 119257e2f323SJérôme Duval else { 119357e2f323SJérôme Duval fTempCount++; 119457e2f323SJérôme Duval if (fTempCount==0) 119557e2f323SJérôme Duval sprintf(name, "Audio Clip"); 119657e2f323SJérôme Duval else 119757e2f323SJérôme Duval sprintf(name, "Audio Clip %d", fTempCount); 119857e2f323SJérôme Duval BPath path; 119957e2f323SJérôme Duval status_t err; 120057e2f323SJérôme Duval BEntry tempEnt; 120157e2f323SJérôme Duval if ((err = fTempDir.GetEntry(&tempEnt)) < B_OK) { 120257e2f323SJérôme Duval return err; 120357e2f323SJérôme Duval } 120457e2f323SJérôme Duval if ((err = tempEnt.GetPath(&path)) < B_OK) { 120557e2f323SJérôme Duval return err; 120657e2f323SJérôme Duval } 120757e2f323SJérôme Duval path.Append(name); 120857e2f323SJérôme Duval int fd; 120957e2f323SJérôme Duval // Use O_EXCL so we know we created the file (sync with other instances) 121057e2f323SJérôme Duval if ((fd = open(path.Path(), O_RDWR | O_CREAT | O_EXCL, 0666)) < 0) { 121157e2f323SJérôme Duval goto again; 121257e2f323SJérôme Duval } 121357e2f323SJérôme Duval close(fd); 121457e2f323SJérôme Duval } 121557e2f323SJérôme Duval return B_OK; 121657e2f323SJérôme Duval } 121757e2f323SJérôme Duval 121857e2f323SJérôme Duval 121957e2f323SJérôme Duval void 122057e2f323SJérôme Duval RecorderWindow::AddSoundItem(const BEntry& entry, bool temp) 122157e2f323SJérôme Duval { 122257e2f323SJérôme Duval // Create list item to display. 122357e2f323SJérôme Duval SoundListItem * listItem = new SoundListItem(entry, temp); 122457e2f323SJérôme Duval fSoundList->AddItem(listItem); 122557e2f323SJérôme Duval fSoundList->Invalidate(); 122657e2f323SJérôme Duval fSoundList->Select(fSoundList->IndexOf(listItem)); 122757e2f323SJérôme Duval } 122857e2f323SJérôme Duval 1229019ed09bSJérôme Duval 1230019ed09bSJérôme Duval void 1231019ed09bSJérôme Duval RecorderWindow::RemoveCurrentSoundItem() { 1232c6f8aa29SJérôme Duval int32 index = fSoundList->CurrentSelection(); 1233c6f8aa29SJérôme Duval BListItem *item = fSoundList->RemoveItem(index); 1234019ed09bSJérôme Duval delete item; 1235c6f8aa29SJérôme Duval if (index >= fSoundList->CountItems()) 1236c6f8aa29SJérôme Duval index = fSoundList->CountItems() - 1; 1237c6f8aa29SJérôme Duval fSoundList->Select(index); 1238019ed09bSJérôme Duval } 1239019ed09bSJérôme Duval 1240019ed09bSJérôme Duval 124157e2f323SJérôme Duval void 1242ef367a8aSStephan Aßmus RecorderWindow::RecordFile(void* cookie, bigtime_t timestamp, 1243ef367a8aSStephan Aßmus void* data, size_t size, const media_raw_audio_format &format) 124457e2f323SJérôme Duval { 124557e2f323SJérôme Duval // Callback called from the SoundConsumer when receiving buffers. 124657e2f323SJérôme Duval RecorderWindow * window = (RecorderWindow *)cookie; 124757e2f323SJérôme Duval 124857e2f323SJérôme Duval if (window->fRecording) { 124957e2f323SJérôme Duval // Write the data to file (we don't buffer or guard file access 125057e2f323SJérôme Duval // or anything) 125157e2f323SJérôme Duval window->fRecFile.WriteAt(window->fRecSize, data, size); 1252ad0c65eaSJérôme Duval window->fVUView->ComputeLevels(data, size, format.format); 125357e2f323SJérôme Duval window->fRecSize += size; 125457e2f323SJérôme Duval } 125557e2f323SJérôme Duval } 125657e2f323SJérôme Duval 125757e2f323SJérôme Duval 125857e2f323SJérôme Duval void 125957e2f323SJérôme Duval RecorderWindow::NotifyRecordFile(void * cookie, int32 code, ...) 126057e2f323SJérôme Duval { 126157e2f323SJérôme Duval if ((code == B_WILL_STOP) || (code == B_NODE_DIES)) { 126257e2f323SJérôme Duval RecorderWindow * window = (RecorderWindow *)cookie; 126357e2f323SJérôme Duval // Tell the window we've stopped, if it doesn't 126457e2f323SJérôme Duval // already know. 126557e2f323SJérôme Duval window->PostMessage(STOP_RECORDING); 126657e2f323SJérôme Duval } 126757e2f323SJérôme Duval } 126857e2f323SJérôme Duval 126957e2f323SJérôme Duval 127057e2f323SJérôme Duval void 1271ad0c65eaSJérôme Duval RecorderWindow::PlayFile(void * cookie, void * data, size_t size, 1272ad0c65eaSJérôme Duval const media_raw_audio_format & format) 127357e2f323SJérôme Duval { 127457e2f323SJérôme Duval // Callback called from the SoundProducer when producing buffers. 127557e2f323SJérôme Duval RecorderWindow * window = (RecorderWindow *)cookie; 127657e2f323SJérôme Duval int32 frame_size = (window->fPlayFormat.u.raw_audio.format & 0xf) * 127757e2f323SJérôme Duval window->fPlayFormat.u.raw_audio.channel_count; 127857e2f323SJérôme Duval 1279948356deSJérôme Duval if ((window->fPlayFrame < window->fPlayLimit) || window->fLooping) { 1280948356deSJérôme Duval if (window->fPlayFrame >= window->fPlayLimit) { 1281948356deSJérôme Duval bigtime_t left = window->fTrackSlider->LeftTime(); 1282948356deSJérôme Duval window->fPlayTrack->SeekToTime(&left); 1283948356deSJérôme Duval window->fPlayFrame = window->fPlayTrack->CurrentFrame(); 1284948356deSJérôme Duval } 128557e2f323SJérôme Duval int64 frames = 0; 128657e2f323SJérôme Duval window->fPlayTrack->ReadFrames(data, &frames); 1287ad0c65eaSJérôme Duval window->fVUView->ComputeLevels(data, size / frame_size, format.format); 128857e2f323SJérôme Duval window->fPlayFrame += size/frame_size; 128957e2f323SJérôme Duval window->PostMessage(UPDATE_TRACKSLIDER); 129057e2f323SJérôme Duval } else { 129157e2f323SJérôme Duval // we're done! 129257e2f323SJérôme Duval window->PostMessage(STOP_PLAYING); 129357e2f323SJérôme Duval } 129457e2f323SJérôme Duval } 129557e2f323SJérôme Duval 12968e32a27eSPhilippe Saint-Pierre 129757e2f323SJérôme Duval void 1298ad0c65eaSJérôme Duval RecorderWindow::NotifyPlayFile(void * cookie, 1299ad0c65eaSJérôme Duval BSoundPlayer::sound_player_notification code, ...) 130057e2f323SJérôme Duval { 130157e2f323SJérôme Duval if ((code == BSoundPlayer::B_STOPPED) || (code == BSoundPlayer::B_SOUND_DONE)) { 130257e2f323SJérôme Duval RecorderWindow * window = (RecorderWindow *)cookie; 130357e2f323SJérôme Duval // tell the window we've stopped, if it doesn't 130457e2f323SJérôme Duval // already know. 130557e2f323SJérôme Duval window->PostMessage(STOP_PLAYING); 130657e2f323SJérôme Duval } 130757e2f323SJérôme Duval } 130857e2f323SJérôme Duval 130957e2f323SJérôme Duval 131057e2f323SJérôme Duval void 131157e2f323SJérôme Duval RecorderWindow::RefsReceived(BMessage *msg) 131257e2f323SJérôme Duval { 131357e2f323SJérôme Duval entry_ref ref; 131457e2f323SJérôme Duval int32 i = 0; 1315c6f8aa29SJérôme Duval int32 countGood = 0; 1316c6f8aa29SJérôme Duval int32 countBad = 0; 131757e2f323SJérôme Duval 131857e2f323SJérôme Duval while (msg->FindRef("refs", i++, &ref) == B_OK) { 131957e2f323SJérôme Duval 132057e2f323SJérôme Duval BEntry entry(&ref, true); 132157e2f323SJérôme Duval BPath path(&entry); 132257e2f323SJérôme Duval BNode node(&entry); 132357e2f323SJérôme Duval 132457e2f323SJérôme Duval if (node.IsFile()) { 1325c6f8aa29SJérôme Duval SoundListItem * listItem = new SoundListItem(entry, false); 1326c6f8aa29SJérôme Duval if (UpdatePlayFile(listItem) == B_OK) { 1327c6f8aa29SJérôme Duval fSoundList->AddItem(listItem); 1328c6f8aa29SJérôme Duval countGood++; 1329c6f8aa29SJérôme Duval continue; 1330c6f8aa29SJérôme Duval } 1331c6f8aa29SJérôme Duval delete listItem; 133257e2f323SJérôme Duval } else if(node.IsDirectory()) { 133357e2f323SJérôme Duval 133457e2f323SJérôme Duval } 1335c6f8aa29SJérôme Duval countBad++; 1336c6f8aa29SJérôme Duval } 1337c6f8aa29SJérôme Duval 1338b1ed9a63SJérôme Duval if (countBad > 0 && countGood == 0) { 1339*aed35104SHumdinger BAlert* alert = new BAlert(B_TRANSLATE("Nothing to play"), 1340*aed35104SHumdinger B_TRANSLATE("None of the files appear to be audio files"), 1341*aed35104SHumdinger B_TRANSLATE("OK"), NULL, NULL, B_WIDTH_AS_USUAL, B_STOP_ALERT); 1342*aed35104SHumdinger alert->SetFlags(alert->Flags() | B_CLOSE_ON_ESCAPE); 1343*aed35104SHumdinger alert->Go(); 1344b1ed9a63SJérôme Duval } else if (countGood > 0) { 1345*aed35104SHumdinger if (countBad > 0) { 1346*aed35104SHumdinger BAlert* alert = new BAlert(B_TRANSLATE("Invalid audio files"), 1347*aed35104SHumdinger B_TRANSLATE("Some of the files don't appear to be audio files"), 13483ee96407SJérôme Duval B_TRANSLATE("OK"), NULL, NULL, B_WIDTH_AS_USUAL, 1349*aed35104SHumdinger B_WARNING_ALERT); 1350*aed35104SHumdinger alert->SetFlags(alert->Flags() | B_CLOSE_ON_ESCAPE); 1351*aed35104SHumdinger alert->Go(); 1352*aed35104SHumdinger } 1353c6f8aa29SJérôme Duval fSoundList->Select(fSoundList->CountItems() - 1); 135457e2f323SJérôme Duval } 135557e2f323SJérôme Duval } 1356019ed09bSJérôme Duval 1357019ed09bSJérôme Duval 1358019ed09bSJérôme Duval void 1359019ed09bSJérôme Duval RecorderWindow::CopyTarget(BMessage *msg) 1360019ed09bSJérôme Duval { 1361019ed09bSJérôme Duval const char *type = NULL; 1362019ed09bSJérôme Duval if (msg->FindString("be:types", &type) == B_OK) { 1363019ed09bSJérôme Duval if (!strcasecmp(type, B_FILE_MIME_TYPE)) { 1364019ed09bSJérôme Duval const char *name; 1365019ed09bSJérôme Duval entry_ref dir; 1366019ed09bSJérôme Duval if (msg->FindString("be:filetypes") == B_OK 1367019ed09bSJérôme Duval && msg->FindString("name", &name) == B_OK 1368019ed09bSJérôme Duval && msg->FindRef("directory", &dir) == B_OK) { 1369019ed09bSJérôme Duval BDirectory directory(&dir); 1370019ed09bSJérôme Duval BFile file(&directory, name, O_RDWR | O_TRUNC); 1371019ed09bSJérôme Duval 1372019ed09bSJérôme Duval // seek time 1373019ed09bSJérôme Duval bigtime_t start = fTrackSlider->LeftTime(); 1374019ed09bSJérôme Duval 1375019ed09bSJérôme Duval // write data 13763ee96407SJérôme Duval bigtime_t diffTime = fTrackSlider->RightTime() 13773ee96407SJérôme Duval - fTrackSlider->LeftTime(); 1378ad0c65eaSJérôme Duval int64 framesToWrite = (int64) (diffTime 1379ad0c65eaSJérôme Duval * fPlayFormat.u.raw_audio.frame_rate / 1000000LL); 1380019ed09bSJérôme Duval int32 frameSize = (fPlayFormat.u.raw_audio.format & 0xf) 1381019ed09bSJérôme Duval * fPlayFormat.u.raw_audio.channel_count; 1382019ed09bSJérôme Duval 1383019ed09bSJérôme Duval wave_struct header; 1384019ed09bSJérôme Duval header.riff.riff_id = FOURCC('R','I','F','F'); 13853ee96407SJérôme Duval header.riff.len 13863ee96407SJérôme Duval = (frameSize * framesToWrite) + sizeof(header) - 8; 1387019ed09bSJérôme Duval header.riff.wave_id = FOURCC('W','A','V','E'); 1388019ed09bSJérôme Duval header.format_chunk.fourcc = FOURCC('f','m','t',' '); 1389019ed09bSJérôme Duval header.format_chunk.len = sizeof(header.format); 1390019ed09bSJérôme Duval header.format.format_tag = 1; 1391019ed09bSJérôme Duval header.format.channels = fPlayFormat.u.raw_audio.channel_count; 13923ee96407SJérôme Duval header.format.samples_per_sec 13933ee96407SJérôme Duval = (uint32)fPlayFormat.u.raw_audio.frame_rate; 13943ee96407SJérôme Duval header.format.avg_bytes_per_sec 13953ee96407SJérôme Duval = (uint32)(fPlayFormat.u.raw_audio.frame_rate 1396019ed09bSJérôme Duval * fPlayFormat.u.raw_audio.channel_count 1397019ed09bSJérôme Duval * (fPlayFormat.u.raw_audio.format & 0xf)); 13983ee96407SJérôme Duval header.format.bits_per_sample 13993ee96407SJérôme Duval = (fPlayFormat.u.raw_audio.format & 0xf) * 8; 1400019ed09bSJérôme Duval header.format.block_align = frameSize; 1401019ed09bSJérôme Duval header.data_chunk.fourcc = FOURCC('d','a','t','a'); 1402019ed09bSJérôme Duval header.data_chunk.len = frameSize * framesToWrite; 1403019ed09bSJérôme Duval file.Seek(0, SEEK_SET); 1404019ed09bSJérôme Duval file.Write(&header, sizeof(header)); 1405019ed09bSJérôme Duval 1406019ed09bSJérôme Duval char *data = (char *)malloc(fPlayFormat.u.raw_audio.buffer_size); 1407019ed09bSJérôme Duval 1408c6f8aa29SJérôme Duval fPlayTrack->SeekToTime(&start); 1409c6f8aa29SJérôme Duval fPlayFrame = fPlayTrack->CurrentFrame(); 1410019ed09bSJérôme Duval while (framesToWrite > 0) { 1411019ed09bSJérôme Duval int64 frames = 0; 1412019ed09bSJérôme Duval status_t err = fPlayTrack->ReadFrames(data, &frames); 1413c6f8aa29SJérôme Duval if (frames <= 0 || err != B_OK) { 1414c6f8aa29SJérôme Duval if (err != B_OK) 1415c6f8aa29SJérôme Duval fprintf(stderr, "CopyTarget: ReadFrames failed\n"); 1416019ed09bSJérôme Duval break; 1417c6f8aa29SJérôme Duval } 1418019ed09bSJérôme Duval file.Write(data, frames * frameSize); 1419019ed09bSJérôme Duval framesToWrite -= frames; 1420019ed09bSJérôme Duval } 1421019ed09bSJérôme Duval 1422019ed09bSJérôme Duval file.Sync(); 1423019ed09bSJérôme Duval free(data); 1424019ed09bSJérôme Duval BNodeInfo nodeInfo(&file); 1425019ed09bSJérôme Duval // set type 1426019ed09bSJérôme Duval } 1427019ed09bSJérôme Duval } else { 1428019ed09bSJérôme Duval 1429019ed09bSJérôme Duval } 1430019ed09bSJérôme Duval } 1431019ed09bSJérôme Duval } 1432