xref: /haiku/src/preferences/media/MediaWindow.cpp (revision 68ea01249e1e2088933cb12f9c28d4e5c5d1c9ef)
1 /*
2  * Copyright 2003-2015, Haiku, Inc.
3  * Distributed under the terms of the MIT license.
4  *
5  * Authors:
6  *		Sikosis, Jérôme Duval
7  *		yourpalal, Alex Wilson
8  */
9 
10 
11 #include "MediaWindow.h"
12 
13 #include <stdio.h>
14 
15 #include <Application.h>
16 #include <Autolock.h>
17 #include <Button.h>
18 #include <CardLayout.h>
19 #include <Catalog.h>
20 #include <Debug.h>
21 #include <Deskbar.h>
22 #include <IconUtils.h>
23 #include <LayoutBuilder.h>
24 #include <Locale.h>
25 #include <MediaRoster.h>
26 #include <MediaTheme.h>
27 #include <Resources.h>
28 #include <Roster.h>
29 #include <Screen.h>
30 #include <ScrollView.h>
31 #include <SeparatorView.h>
32 #include <SpaceLayoutItem.h>
33 #include <StorageKit.h>
34 #include <String.h>
35 #include <TextView.h>
36 
37 #include "Media.h"
38 #include "MediaIcons.h"
39 #include "MidiSettingsView.h"
40 
41 #undef B_TRANSLATION_CONTEXT
42 #define B_TRANSLATION_CONTEXT "Media Window"
43 
44 
45 const uint32 ML_SELECTED_NODE = 'MlSN';
46 const uint32 ML_RESTART_THREAD_FINISHED = 'MlRF';
47 
48 
49 class NodeListItemUpdater : public MediaListItem::Visitor {
50 public:
51 	typedef void (NodeListItem::*UpdateMethod)(bool);
52 
53 	NodeListItemUpdater(NodeListItem* target, UpdateMethod action)
54 		:
55 		fComparator(target),
56 		fAction(action)
57 	{
58 	}
59 
60 
61 	virtual	void	Visit(AudioMixerListItem*){}
62 	virtual	void	Visit(DeviceListItem*){}
63 	virtual	void	Visit(MidiListItem*){}
64 	virtual void	Visit(NodeListItem* item)
65 	{
66 		item->Accept(fComparator);
67 		(item->*(fAction))(fComparator.result == 0);
68 	}
69 
70 private:
71 
72 			NodeListItem::Comparator		fComparator;
73 			UpdateMethod					fAction;
74 };
75 
76 
77 MediaWindow::SmartNode::SmartNode(const BMessenger& notifyHandler)
78 	:
79 	fNode(NULL),
80 	fMessenger(notifyHandler)
81 {
82 }
83 
84 
85 MediaWindow::SmartNode::~SmartNode()
86 {
87 	_FreeNode();
88 }
89 
90 
91 void
92 MediaWindow::SmartNode::SetTo(const dormant_node_info* info)
93 {
94 	_FreeNode();
95 	if (!info)
96 		return;
97 
98 	fNode = new media_node();
99 	BMediaRoster* roster = BMediaRoster::Roster();
100 
101 	status_t status = B_OK;
102 	media_node_id node_id;
103 	if (roster->GetInstancesFor(info->addon, info->flavor_id, &node_id) == B_OK)
104 		status = roster->GetNodeFor(node_id, fNode);
105 	else
106 		status = roster->InstantiateDormantNode(*info, fNode, B_FLAVOR_IS_GLOBAL);
107 
108 	if (status != B_OK) {
109 		fprintf(stderr, "SmartNode::SetTo error with node %" B_PRId32
110 			": %s\n", fNode->node, strerror(status));
111 	}
112 
113 	status = roster->StartWatching(fMessenger, *fNode, B_MEDIA_WILDCARD);
114 	if (status != B_OK) {
115 		fprintf(stderr, "SmartNode::SetTo can't start watching for"
116 			" node %" B_PRId32 "\n", fNode->node);
117 	}
118 }
119 
120 
121 void
122 MediaWindow::SmartNode::SetTo(const media_node& node)
123 {
124 	_FreeNode();
125 	fNode = new media_node(node);
126 	BMediaRoster* roster = BMediaRoster::Roster();
127 	roster->StartWatching(fMessenger, *fNode, B_MEDIA_WILDCARD);
128 }
129 
130 
131 bool
132 MediaWindow::SmartNode::IsSet()
133 {
134 	return fNode != NULL;
135 }
136 
137 
138 MediaWindow::SmartNode::operator media_node()
139 {
140 	if (fNode)
141 		return *fNode;
142 	media_node node;
143 	return node;
144 }
145 
146 
147 void
148 MediaWindow::SmartNode::_FreeNode()
149 {
150 	if (!IsSet())
151 		return;
152 
153 	BMediaRoster* roster = BMediaRoster::Roster();
154 	if (roster != NULL) {
155 		status_t status = roster->StopWatching(fMessenger,
156 			*fNode, B_MEDIA_WILDCARD);
157 		if (status != B_OK) {
158 			fprintf(stderr, "SmartNode::_FreeNode can't unwatch"
159 				" media services for node %" B_PRId32 "\n", fNode->node);
160 		}
161 
162 		roster->ReleaseNode(*fNode);
163 		if (status != B_OK) {
164 			fprintf(stderr, "SmartNode::_FreeNode can't release"
165 				" node %" B_PRId32 "\n", fNode->node);
166 		}
167 	}
168 	delete fNode;
169 	fNode = NULL;
170 }
171 
172 
173 // #pragma mark -
174 
175 
176 MediaWindow::MediaWindow(BRect frame)
177 	:
178 	BWindow(frame, B_TRANSLATE_SYSTEM_NAME("Media"), B_TITLED_WINDOW,
179 		B_ASYNCHRONOUS_CONTROLS | B_AUTO_UPDATE_SIZE_LIMITS),
180 	fCurrentNode(BMessenger(this)),
181 	fParamWeb(NULL),
182 	fAudioInputs(5, true),
183 	fAudioOutputs(5, true),
184 	fVideoInputs(5, true),
185 	fVideoOutputs(5, true),
186 	fInitCheck(B_OK),
187 	fRestartThread(-1),
188 	fRestartAlert(NULL)
189 {
190 	_InitWindow();
191 
192 	BMediaRoster* roster = BMediaRoster::Roster();
193 	roster->StartWatching(BMessenger(this, this),
194 		B_MEDIA_SERVER_STARTED);
195 	roster->StartWatching(BMessenger(this, this),
196 		B_MEDIA_SERVER_QUIT);
197 }
198 
199 
200 MediaWindow::~MediaWindow()
201 {
202 	_EmptyNodeLists();
203 	_ClearParamView();
204 
205 	char buffer[512];
206 	BRect rect = Frame();
207 	PRINT_OBJECT(rect);
208 	snprintf(buffer, 512, "# MediaPrefs Settings\n rect = %i,%i,%i,%i\n",
209 		int(rect.left), int(rect.top), int(rect.right), int(rect.bottom));
210 
211 	BPath path;
212 	if (find_directory(B_USER_SETTINGS_DIRECTORY, &path) == B_OK) {
213 		path.Append(SETTINGS_FILE);
214 		BFile file(path.Path(), B_READ_WRITE | B_CREATE_FILE | B_ERASE_FILE);
215 		if (file.InitCheck() == B_OK)
216 			file.Write(buffer, strlen(buffer));
217 	}
218 
219 	BMediaRoster* roster = BMediaRoster::CurrentRoster();
220 	roster->StopWatching(BMessenger(this, this),
221 		B_MEDIA_SERVER_STARTED);
222 	roster->StartWatching(BMessenger(this, this),
223 		B_MEDIA_SERVER_QUIT);
224 }
225 
226 
227 status_t
228 MediaWindow::InitCheck()
229 {
230 	return fInitCheck;
231 }
232 
233 
234 void
235 MediaWindow::SelectNode(const dormant_node_info* node)
236 {
237 	fCurrentNode.SetTo(node);
238 	_MakeParamView();
239 	fTitleView->SetLabel(node->name);
240 }
241 
242 
243 void
244 MediaWindow::SelectAudioSettings(const char* title)
245 {
246 	fContentLayout->SetVisibleItem(fContentLayout->IndexOfView(fAudioView));
247 	fTitleView->SetLabel(title);
248 }
249 
250 
251 void
252 MediaWindow::SelectVideoSettings(const char* title)
253 {
254 	fContentLayout->SetVisibleItem(fContentLayout->IndexOfView(fVideoView));
255 	fTitleView->SetLabel(title);
256 }
257 
258 
259 void
260 MediaWindow::SelectAudioMixer(const char* title)
261 {
262 	media_node mixerNode;
263 	BMediaRoster* roster = BMediaRoster::Roster();
264 	roster->GetAudioMixer(&mixerNode);
265 	fCurrentNode.SetTo(mixerNode);
266 	_MakeParamView();
267 	fTitleView->SetLabel(title);
268 }
269 
270 
271 void
272 MediaWindow::SelectMidiSettings(const char* title)
273 {
274 	fContentLayout->SetVisibleItem(fContentLayout->IndexOfView(fMidiView));
275 	fTitleView->SetLabel(title);
276 }
277 
278 
279 void
280 MediaWindow::UpdateInputListItem(MediaListItem::media_type type,
281 	const dormant_node_info* node)
282 {
283 	NodeListItem compareTo(node, type);
284 	NodeListItemUpdater updater(&compareTo, &NodeListItem::SetDefaultInput);
285 	for (int32 i = 0; i < fListView->CountItems(); i++) {
286 		MediaListItem* item = static_cast<MediaListItem*>(fListView->ItemAt(i));
287 		item->Accept(updater);
288 	}
289 	fListView->Invalidate();
290 }
291 
292 
293 void
294 MediaWindow::UpdateOutputListItem(MediaListItem::media_type type,
295 	const dormant_node_info* node)
296 {
297 	NodeListItem compareTo(node, type);
298 	NodeListItemUpdater updater(&compareTo, &NodeListItem::SetDefaultOutput);
299 	for (int32 i = 0; i < fListView->CountItems(); i++) {
300 		MediaListItem* item = static_cast<MediaListItem*>(fListView->ItemAt(i));
301 		item->Accept(updater);
302 	}
303 	fListView->Invalidate();
304 }
305 
306 
307 bool
308 MediaWindow::QuitRequested()
309 {
310 	if (fRestartThread > 0) {
311 		BString text(B_TRANSLATE("Quitting Media now will stop the "
312 			"restarting of the media services. Flaky or unavailable media "
313 			"functionality is the likely result."));
314 
315 		fRestartAlert = new BAlert(B_TRANSLATE("Warning!"), text,
316 			B_TRANSLATE("Quit anyway"), NULL, NULL,
317 			B_WIDTH_AS_USUAL, B_OFFSET_SPACING, B_WARNING_ALERT);
318 
319 		fRestartAlert->Go();
320 	}
321 	// Stop watching the MediaRoster
322 	fCurrentNode.SetTo(NULL);
323 	be_app->PostMessage(B_QUIT_REQUESTED);
324 	return true;
325 }
326 
327 
328 void
329 MediaWindow::MessageReceived(BMessage* message)
330 {
331 	switch (message->what) {
332 		case ML_RESTART_THREAD_FINISHED:
333 			fRestartThread = -1;
334 			_InitMedia(false);
335 			break;
336 
337 		case ML_RESTART_MEDIA_SERVER:
338 		{
339 			fRestartThread = spawn_thread(&MediaWindow::_RestartMediaServices,
340 				"restart_thread", B_NORMAL_PRIORITY, this);
341 			if (fRestartThread < 0)
342 				fprintf(stderr, "couldn't create restart thread\n");
343 			else
344 				resume_thread(fRestartThread);
345 			break;
346 		}
347 
348 		case B_MEDIA_WEB_CHANGED:
349 		case ML_SELECTED_NODE:
350 		{
351 			PRINT_OBJECT(*message);
352 
353 			MediaListItem* item = static_cast<MediaListItem*>(
354 					fListView->ItemAt(fListView->CurrentSelection()));
355 			if (item == NULL)
356 				break;
357 
358 			fCurrentNode.SetTo(NULL);
359 			_ClearParamView();
360 			item->AlterWindow(this);
361 			break;
362 		}
363 
364 		case B_MEDIA_SERVER_STARTED:
365 		case B_MEDIA_SERVER_QUIT:
366 		{
367 			PRINT_OBJECT(*message);
368 			_InitMedia(false);
369 			break;
370 		}
371 
372 		default:
373 			BWindow::MessageReceived(message);
374 			break;
375 	}
376 }
377 
378 
379 // #pragma mark - private
380 
381 
382 void
383 MediaWindow::_InitWindow()
384 {
385 	fListView = new BListView("media_list_view");
386 	fListView->SetSelectionMessage(new BMessage(ML_SELECTED_NODE));
387 	fListView->SetExplicitMinSize(BSize(140, B_SIZE_UNSET));
388 
389 	// Add ScrollView to Media Menu for pretty border
390 	BScrollView* scrollView = new BScrollView("listscroller",
391 		fListView, 0, false, false, B_FANCY_BORDER);
392 
393 	// Create the Views
394 	fTitleView = new BSeparatorView(B_HORIZONTAL, B_FANCY_BORDER);
395 	fTitleView->SetLabel(B_TRANSLATE("Audio settings"));
396 	fTitleView->SetFont(be_bold_font);
397 
398 	fContentLayout = new BCardLayout();
399 	new BView("content view", 0, fContentLayout);
400 	fContentLayout->Owner()->SetViewUIColor(B_PANEL_BACKGROUND_COLOR);
401 	fContentLayout->SetExplicitMaxSize(BSize(B_SIZE_UNLIMITED, B_SIZE_UNSET));
402 
403 	fAudioView = new AudioSettingsView();
404 	fContentLayout->AddView(fAudioView);
405 
406 	fVideoView = new VideoSettingsView();
407 	fContentLayout->AddView(fVideoView);
408 
409 	fMidiView = new MidiSettingsView();
410 	fContentLayout->AddView(fMidiView);
411 
412 	// Layout all views
413 	BLayoutBuilder::Group<>(this, B_HORIZONTAL)
414 		.SetInsets(B_USE_WINDOW_SPACING)
415 		.Add(scrollView, 0.0f)
416 		.AddGroup(B_VERTICAL)
417 			.SetInsets(0, 0, 0, 0)
418 			.Add(fTitleView)
419 			.Add(fContentLayout);
420 
421 	// Start the window
422 	fInitCheck = _InitMedia(true);
423 	if (fInitCheck != B_OK)
424 		PostMessage(B_QUIT_REQUESTED);
425 	else if (IsHidden())
426 		Show();
427 }
428 
429 
430 status_t
431 MediaWindow::_InitMedia(bool first)
432 {
433 	status_t err = B_OK;
434 	BMediaRoster* roster = BMediaRoster::Roster(&err);
435 
436 	if (first && err != B_OK) {
437 		BAlert* alert = new BAlert("start_media_server",
438 			B_TRANSLATE("Could not connect to the media server.\n"
439 				"Would you like to start it ?"),
440 			B_TRANSLATE("Quit"),
441 			B_TRANSLATE("Start media server"), NULL,
442 			B_WIDTH_AS_USUAL, B_WARNING_ALERT);
443 		alert->SetShortcut(0, B_ESCAPE);
444 		if (alert->Go() == 0)
445 			return B_ERROR;
446 
447 		Show();
448 
449 		launch_media_server();
450 	}
451 
452 	Lock();
453 
454 	bool isVideoSelected = true;
455 	if (!first && fListView->ItemAt(0) != NULL
456 		&& fListView->ItemAt(0)->IsSelected())
457 		isVideoSelected = false;
458 
459 	while (fListView->CountItems() > 0)
460 		delete fListView->RemoveItem((int32)0);
461 	_EmptyNodeLists();
462 
463 	// Grab Media Info
464 	_FindNodes();
465 
466 	// Add video nodes first. They might have an additional audio
467 	// output or input, but still should be listed as video node.
468 	_AddNodeItems(fVideoOutputs, MediaListItem::VIDEO_TYPE);
469 	_AddNodeItems(fVideoInputs, MediaListItem::VIDEO_TYPE);
470 	_AddNodeItems(fAudioOutputs, MediaListItem::AUDIO_TYPE);
471 	_AddNodeItems(fAudioInputs, MediaListItem::AUDIO_TYPE);
472 
473 	fAudioView->AddOutputNodes(fAudioOutputs);
474 	fAudioView->AddInputNodes(fAudioInputs);
475 	fVideoView->AddOutputNodes(fVideoOutputs);
476 	fVideoView->AddInputNodes(fVideoInputs);
477 
478 	// build our list view
479 	DeviceListItem* audio = new DeviceListItem(B_TRANSLATE("Audio settings"),
480 		MediaListItem::AUDIO_TYPE);
481 	fListView->AddItem(audio);
482 
483 	MidiListItem* midi = new MidiListItem(B_TRANSLATE("MIDI Settings"));
484 	fListView->AddItem(midi);
485 
486 	MediaListItem* video = new DeviceListItem(B_TRANSLATE("Video settings"),
487 		MediaListItem::VIDEO_TYPE);
488 	fListView->AddItem(video);
489 
490 	MediaListItem* mixer = new AudioMixerListItem(B_TRANSLATE("Audio mixer"));
491 	fListView->AddItem(mixer);
492 
493 	fListView->SortItems(&MediaListItem::Compare);
494 	_UpdateListViewMinWidth();
495 
496 	// Set default nodes for our setting views
497 	media_node defaultNode;
498 	dormant_node_info nodeInfo;
499 	int32 outputID;
500 	BString outputName;
501 
502 	if (roster->GetAudioInput(&defaultNode) == B_OK) {
503 		roster->GetDormantNodeFor(defaultNode, &nodeInfo);
504 		fAudioView->SetDefaultInput(&nodeInfo);
505 			// this causes our listview to be updated as well
506 	}
507 
508 	if (roster->GetAudioOutput(&defaultNode, &outputID, &outputName) == B_OK) {
509 		roster->GetDormantNodeFor(defaultNode, &nodeInfo);
510 		fAudioView->SetDefaultOutput(&nodeInfo);
511 		fAudioView->SetDefaultChannel(outputID);
512 			// this causes our listview to be updated as well
513 	}
514 
515 	if (roster->GetVideoInput(&defaultNode) == B_OK) {
516 		roster->GetDormantNodeFor(defaultNode, &nodeInfo);
517 		fVideoView->SetDefaultInput(&nodeInfo);
518 			// this causes our listview to be updated as well
519 	}
520 
521 	if (roster->GetVideoOutput(&defaultNode) == B_OK) {
522 		roster->GetDormantNodeFor(defaultNode, &nodeInfo);
523 		fVideoView->SetDefaultOutput(&nodeInfo);
524 			// this causes our listview to be updated as well
525 	}
526 
527 	if (first)
528 		fListView->Select(fListView->IndexOf(mixer));
529 	else if (isVideoSelected)
530 		fListView->Select(fListView->IndexOf(video));
531 	else
532 		fListView->Select(fListView->IndexOf(audio));
533 
534 	Unlock();
535 
536 	return B_OK;
537 }
538 
539 
540 void
541 MediaWindow::_FindNodes()
542 {
543 	_FindNodes(B_MEDIA_RAW_AUDIO, B_PHYSICAL_OUTPUT, fAudioOutputs);
544 	_FindNodes(B_MEDIA_RAW_AUDIO, B_PHYSICAL_INPUT, fAudioInputs);
545 	_FindNodes(B_MEDIA_ENCODED_AUDIO, B_PHYSICAL_OUTPUT, fAudioOutputs);
546 	_FindNodes(B_MEDIA_ENCODED_AUDIO, B_PHYSICAL_INPUT, fAudioInputs);
547 	_FindNodes(B_MEDIA_RAW_VIDEO, B_PHYSICAL_OUTPUT, fVideoOutputs);
548 	_FindNodes(B_MEDIA_RAW_VIDEO, B_PHYSICAL_INPUT, fVideoInputs);
549 	_FindNodes(B_MEDIA_ENCODED_VIDEO, B_PHYSICAL_OUTPUT, fVideoOutputs);
550 	_FindNodes(B_MEDIA_ENCODED_VIDEO, B_PHYSICAL_INPUT, fVideoInputs);
551 }
552 
553 
554 void
555 MediaWindow::_FindNodes(media_type type, uint64 kind, NodeList& into)
556 {
557 	dormant_node_info nodeInfo[64];
558 	int32 nodeInfoCount = 64;
559 
560 	media_format format;
561 	media_format* nodeInputFormat = NULL;
562 	media_format* nodeOutputFormat = NULL;
563 	format.type = type;
564 
565 	// output nodes must be BBufferConsumers => they have an input format
566 	// input nodes must be BBufferProducers => they have an output format
567 	if ((kind & B_PHYSICAL_OUTPUT) != 0)
568 		nodeInputFormat = &format;
569 	else if ((kind & B_PHYSICAL_INPUT) != 0)
570 		nodeOutputFormat = &format;
571 	else
572 		return;
573 
574 	BMediaRoster* roster = BMediaRoster::Roster();
575 
576 	if (roster->GetDormantNodes(nodeInfo, &nodeInfoCount, nodeInputFormat,
577 			nodeOutputFormat, NULL, kind) != B_OK) {
578 		// TODO: better error reporting!
579 		fprintf(stderr, "error\n");
580 		return;
581 	}
582 
583 	for (int32 i = 0; i < nodeInfoCount; i++) {
584 		PRINT(("node : %s, media_addon %i, flavor_id %i\n",
585 			nodeInfo[i].name, (int)nodeInfo[i].addon,
586 			(int)nodeInfo[i].flavor_id));
587 
588 		dormant_node_info* info = new dormant_node_info();
589 		strlcpy(info->name, nodeInfo[i].name, B_MEDIA_NAME_LENGTH);
590 		info->flavor_id = nodeInfo[i].flavor_id;
591 		info->addon = nodeInfo[i].addon;
592 		into.AddItem(info);
593 	}
594 }
595 
596 
597 void
598 MediaWindow::_AddNodeItems(NodeList& list, MediaListItem::media_type type)
599 {
600 	int32 count = list.CountItems();
601 	for (int32 i = 0; i < count; i++) {
602 		dormant_node_info* info = list.ItemAt(i);
603 		if (_FindNodeListItem(info) == NULL)
604 			fListView->AddItem(new NodeListItem(info, type));
605 	}
606 }
607 
608 
609 void
610 MediaWindow::_EmptyNodeLists()
611 {
612 	fAudioOutputs.MakeEmpty();
613 	fAudioInputs.MakeEmpty();
614 	fVideoOutputs.MakeEmpty();
615 	fVideoInputs.MakeEmpty();
616 }
617 
618 
619 NodeListItem*
620 MediaWindow::_FindNodeListItem(dormant_node_info* info)
621 {
622 	NodeListItem audioItem(info, MediaListItem::AUDIO_TYPE);
623 	NodeListItem videoItem(info, MediaListItem::VIDEO_TYPE);
624 
625 	NodeListItem::Comparator audioComparator(&audioItem);
626 	NodeListItem::Comparator videoComparator(&videoItem);
627 
628 	for (int32 i = 0; i < fListView->CountItems(); i++) {
629 		MediaListItem* item = static_cast<MediaListItem*>(fListView->ItemAt(i));
630 		item->Accept(audioComparator);
631 		if (audioComparator.result == 0)
632 			return static_cast<NodeListItem*>(item);
633 
634 		item->Accept(videoComparator);
635 		if (videoComparator.result == 0)
636 			return static_cast<NodeListItem*>(item);
637 	}
638 	return NULL;
639 }
640 
641 
642 void
643 MediaWindow::_UpdateListViewMinWidth()
644 {
645 	float width = 0;
646 	for (int32 i = 0; i < fListView->CountItems(); i++) {
647 		BListItem* item = fListView->ItemAt(i);
648 		width = max_c(width, item->Width());
649 	}
650 	fListView->SetExplicitMinSize(BSize(width, B_SIZE_UNSET));
651 	fListView->InvalidateLayout();
652 }
653 
654 
655 status_t
656 MediaWindow::_RestartMediaServices(void* data)
657 {
658 	MediaWindow* window = (MediaWindow*)data;
659 
660 	shutdown_media_server();
661 
662 	if (window->fRestartAlert != NULL
663 			&& window->fRestartAlert->Lock()) {
664 		window->fRestartAlert->Quit();
665 	}
666 
667 	return window->PostMessage(ML_RESTART_THREAD_FINISHED);
668 }
669 
670 
671 void
672 MediaWindow::_ClearParamView()
673 {
674 	BLayoutItem* item = fContentLayout->VisibleItem();
675 	if (!item)
676 		return;
677 
678 	BView* view = item->View();
679 	if (view != fVideoView && view != fAudioView && view != fMidiView) {
680 		fContentLayout->RemoveItem(item);
681 		delete view;
682 		delete fParamWeb;
683 		fParamWeb = NULL;
684 	}
685 }
686 
687 
688 void
689 MediaWindow::_MakeParamView()
690 {
691 	if (!fCurrentNode.IsSet())
692 		return;
693 
694 	fParamWeb = NULL;
695 	BMediaRoster* roster = BMediaRoster::Roster();
696 	if (roster->GetParameterWebFor(fCurrentNode, &fParamWeb) == B_OK) {
697 		BRect hint(fContentLayout->Frame());
698 		BView* paramView = BMediaTheme::ViewFor(fParamWeb, &hint);
699 		if (paramView) {
700 			fContentLayout->AddView(paramView);
701 			fContentLayout->SetVisibleItem(fContentLayout->CountItems() - 1);
702 			return;
703 		}
704 	}
705 
706 	_MakeEmptyParamView();
707 }
708 
709 
710 void
711 MediaWindow::_MakeEmptyParamView()
712 {
713 	fParamWeb = NULL;
714 
715 	BStringView* stringView = new BStringView("noControls",
716 		B_TRANSLATE("This hardware has no controls."));
717 
718 	BSize unlimited(B_SIZE_UNLIMITED, B_SIZE_UNLIMITED);
719 	stringView->SetExplicitMaxSize(unlimited);
720 
721 	BAlignment centered(B_ALIGN_HORIZONTAL_CENTER,
722 		B_ALIGN_VERTICAL_CENTER);
723 	stringView->SetExplicitAlignment(centered);
724 	stringView->SetAlignment(B_ALIGN_CENTER);
725 
726 	fContentLayout->AddView(stringView);
727 	fContentLayout->SetVisibleItem(fContentLayout->CountItems() - 1);
728 
729 	rgb_color panel = stringView->LowColor();
730 	stringView->SetHighColor(tint_color(panel,
731 		B_DISABLED_LABEL_TINT));
732 }
733 
734