2015-12-02 06:04:04 +00:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2014 Jared Boone, ShareBrained Technology, Inc.
|
2018-05-21 17:46:48 +00:00
|
|
|
* Copyright (C) 2018 Furrtek
|
2015-12-02 06:04:04 +00:00
|
|
|
*
|
|
|
|
* This file is part of PortaPack.
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 2, or (at your option)
|
|
|
|
* any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; see the file COPYING. If not, write to
|
|
|
|
* the Free Software Foundation, Inc., 51 Franklin Street,
|
|
|
|
* Boston, MA 02110-1301, USA.
|
|
|
|
*/
|
|
|
|
|
2015-12-08 23:28:33 +00:00
|
|
|
#include "analog_audio_app.hpp"
|
2015-12-02 06:04:04 +00:00
|
|
|
|
2016-06-24 18:30:54 +00:00
|
|
|
#include "baseband_api.hpp"
|
|
|
|
|
2015-12-02 06:04:04 +00:00
|
|
|
#include "portapack.hpp"
|
2016-02-10 23:34:33 +00:00
|
|
|
#include "portapack_persistent_memory.hpp"
|
2015-12-02 06:04:04 +00:00
|
|
|
using namespace portapack;
|
2017-11-28 07:52:04 +00:00
|
|
|
using namespace tonekey;
|
2015-12-02 06:04:04 +00:00
|
|
|
|
2016-02-06 00:25:43 +00:00
|
|
|
#include "audio.hpp"
|
2016-04-20 17:22:50 +00:00
|
|
|
#include "file.hpp"
|
2016-02-05 23:22:28 +00:00
|
|
|
|
2015-12-02 06:04:04 +00:00
|
|
|
#include "utility.hpp"
|
|
|
|
|
2016-04-30 18:25:04 +00:00
|
|
|
#include "string_format.hpp"
|
|
|
|
|
2016-01-31 17:13:44 +00:00
|
|
|
namespace ui {
|
|
|
|
|
2016-02-02 21:42:00 +00:00
|
|
|
/* AMOptionsView *********************************************************/
|
|
|
|
|
|
|
|
AMOptionsView::AMOptionsView(
|
|
|
|
const Rect parent_rect, const Style* const style
|
|
|
|
) : View { parent_rect }
|
|
|
|
{
|
|
|
|
set_style(style);
|
|
|
|
|
2016-09-05 21:53:04 +00:00
|
|
|
add_children({
|
2016-02-02 21:42:00 +00:00
|
|
|
&label_config,
|
|
|
|
&options_config,
|
2016-09-05 21:53:04 +00:00
|
|
|
});
|
2016-02-02 21:42:00 +00:00
|
|
|
|
2016-02-02 23:20:14 +00:00
|
|
|
options_config.set_selected_index(receiver_model.am_configuration());
|
2016-02-02 21:42:00 +00:00
|
|
|
options_config.on_change = [this](size_t n, OptionsField::value_t) {
|
2016-02-02 23:20:14 +00:00
|
|
|
receiver_model.set_am_configuration(n);
|
2016-02-02 21:42:00 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
/* NBFMOptionsView *******************************************************/
|
|
|
|
|
|
|
|
NBFMOptionsView::NBFMOptionsView(
|
|
|
|
const Rect parent_rect, const Style* const style
|
|
|
|
) : View { parent_rect }
|
|
|
|
{
|
|
|
|
set_style(style);
|
|
|
|
|
2016-09-05 21:53:04 +00:00
|
|
|
add_children({
|
2016-02-02 21:42:00 +00:00
|
|
|
&label_config,
|
|
|
|
&options_config,
|
2017-06-11 08:50:29 +00:00
|
|
|
&text_squelch,
|
|
|
|
&field_squelch
|
2016-09-05 21:53:04 +00:00
|
|
|
});
|
2016-02-02 21:42:00 +00:00
|
|
|
|
2016-02-02 23:20:14 +00:00
|
|
|
options_config.set_selected_index(receiver_model.nbfm_configuration());
|
2016-02-02 21:42:00 +00:00
|
|
|
options_config.on_change = [this](size_t n, OptionsField::value_t) {
|
2016-02-02 23:20:14 +00:00
|
|
|
receiver_model.set_nbfm_configuration(n);
|
2016-02-02 21:42:00 +00:00
|
|
|
};
|
2017-06-11 08:50:29 +00:00
|
|
|
|
|
|
|
field_squelch.set_value(receiver_model.squelch_level());
|
|
|
|
field_squelch.on_change = [this](int32_t v) {
|
|
|
|
receiver_model.set_squelch_level(v);
|
|
|
|
};
|
2016-02-02 21:42:00 +00:00
|
|
|
}
|
|
|
|
|
2016-01-31 17:13:44 +00:00
|
|
|
/* AnalogAudioView *******************************************************/
|
|
|
|
|
|
|
|
AnalogAudioView::AnalogAudioView(
|
2018-06-12 06:55:12 +00:00
|
|
|
NavigationView& nav
|
2017-05-18 10:06:11 +00:00
|
|
|
) : nav_ (nav)
|
|
|
|
{
|
2016-09-05 21:53:04 +00:00
|
|
|
add_children({
|
2016-01-31 17:13:44 +00:00
|
|
|
&rssi,
|
|
|
|
&channel,
|
|
|
|
&audio,
|
|
|
|
&field_frequency,
|
|
|
|
&field_lna,
|
|
|
|
&field_vga,
|
|
|
|
&options_modulation,
|
|
|
|
&field_volume,
|
2017-11-28 07:52:04 +00:00
|
|
|
&text_ctcss,
|
2016-04-30 20:56:54 +00:00
|
|
|
&record_view,
|
2018-05-21 17:46:48 +00:00
|
|
|
&waterfall
|
2016-09-05 21:53:04 +00:00
|
|
|
});
|
2016-01-31 17:13:44 +00:00
|
|
|
|
|
|
|
field_frequency.set_value(receiver_model.tuning_frequency());
|
|
|
|
field_frequency.set_step(receiver_model.frequency_step());
|
|
|
|
field_frequency.on_change = [this](rf::Frequency f) {
|
|
|
|
this->on_tuning_frequency_changed(f);
|
|
|
|
};
|
|
|
|
field_frequency.on_edit = [this, &nav]() {
|
|
|
|
// TODO: Provide separate modal method/scheme?
|
|
|
|
auto new_view = nav.push<FrequencyKeypadView>(receiver_model.tuning_frequency());
|
|
|
|
new_view->on_changed = [this](rf::Frequency f) {
|
|
|
|
this->on_tuning_frequency_changed(f);
|
|
|
|
this->field_frequency.set_value(f);
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
field_frequency.on_show_options = [this]() {
|
|
|
|
this->on_show_options_frequency();
|
|
|
|
};
|
|
|
|
|
|
|
|
field_lna.on_show_options = [this]() {
|
|
|
|
this->on_show_options_rf_gain();
|
|
|
|
};
|
|
|
|
|
2016-04-12 17:49:07 +00:00
|
|
|
field_vga.on_show_options = [this]() {
|
|
|
|
this->on_show_options_rf_gain();
|
|
|
|
};
|
2016-01-31 17:13:44 +00:00
|
|
|
|
2016-02-02 23:01:04 +00:00
|
|
|
const auto modulation = receiver_model.modulation();
|
2016-07-27 21:51:37 +00:00
|
|
|
options_modulation.set_by_value(toUType(modulation));
|
2016-02-01 17:12:55 +00:00
|
|
|
options_modulation.on_change = [this](size_t, OptionsField::value_t v) {
|
|
|
|
this->on_modulation_changed(static_cast<ReceiverModel::Mode>(v));
|
2016-01-31 17:13:44 +00:00
|
|
|
};
|
2016-02-02 21:42:00 +00:00
|
|
|
options_modulation.on_show_options = [this]() {
|
|
|
|
this->on_show_options_modulation();
|
|
|
|
};
|
2016-01-31 17:13:44 +00:00
|
|
|
|
2016-02-06 00:25:43 +00:00
|
|
|
field_volume.set_value((receiver_model.headphone_volume() - audio::headphone::volume_range().max).decibel() + 99);
|
2016-01-31 17:13:44 +00:00
|
|
|
field_volume.on_change = [this](int32_t v) {
|
|
|
|
this->on_headphone_volume_changed(v);
|
|
|
|
};
|
|
|
|
|
2016-05-13 05:00:40 +00:00
|
|
|
record_view.on_error = [&nav](std::string message) {
|
2016-05-17 17:15:34 +00:00
|
|
|
nav.display_modal("Error", message);
|
2016-05-13 05:00:40 +00:00
|
|
|
};
|
2018-04-18 21:44:41 +00:00
|
|
|
|
|
|
|
waterfall.on_select = [this](int32_t offset) {
|
|
|
|
field_frequency.set_value(receiver_model.tuning_frequency() + offset);
|
|
|
|
};
|
2016-05-13 05:00:40 +00:00
|
|
|
|
2016-02-06 00:25:43 +00:00
|
|
|
audio::output::start();
|
|
|
|
|
2016-02-02 23:01:04 +00:00
|
|
|
update_modulation(static_cast<ReceiverModel::Mode>(modulation));
|
2016-01-31 17:13:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
AnalogAudioView::~AnalogAudioView() {
|
|
|
|
// TODO: Manipulating audio codec here, and in ui_receiver.cpp. Good to do
|
|
|
|
// both?
|
2016-02-06 00:25:43 +00:00
|
|
|
audio::output::stop();
|
2016-01-31 17:13:44 +00:00
|
|
|
|
|
|
|
receiver_model.disable();
|
2016-06-24 18:30:54 +00:00
|
|
|
|
|
|
|
baseband::shutdown();
|
2016-01-31 17:13:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void AnalogAudioView::on_hide() {
|
|
|
|
// TODO: Terrible kludge because widget system doesn't notify Waterfall that
|
|
|
|
// it's being shown or hidden.
|
|
|
|
waterfall.on_hide();
|
|
|
|
View::on_hide();
|
|
|
|
}
|
|
|
|
|
|
|
|
void AnalogAudioView::set_parent_rect(const Rect new_parent_rect) {
|
|
|
|
View::set_parent_rect(new_parent_rect);
|
2018-05-21 17:46:48 +00:00
|
|
|
|
2016-11-29 18:13:56 +00:00
|
|
|
const ui::Rect waterfall_rect { 0, header_height, new_parent_rect.width(), new_parent_rect.height() - header_height };
|
2016-01-31 17:13:44 +00:00
|
|
|
waterfall.set_parent_rect(waterfall_rect);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AnalogAudioView::focus() {
|
|
|
|
field_frequency.focus();
|
|
|
|
}
|
|
|
|
|
|
|
|
void AnalogAudioView::on_tuning_frequency_changed(rf::Frequency f) {
|
|
|
|
receiver_model.set_tuning_frequency(f);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AnalogAudioView::on_baseband_bandwidth_changed(uint32_t bandwidth_hz) {
|
|
|
|
receiver_model.set_baseband_bandwidth(bandwidth_hz);
|
|
|
|
}
|
|
|
|
|
2016-02-02 21:39:02 +00:00
|
|
|
void AnalogAudioView::on_modulation_changed(const ReceiverModel::Mode modulation) {
|
2016-01-31 17:13:44 +00:00
|
|
|
// TODO: Terrible kludge because widget system doesn't notify Waterfall that
|
|
|
|
// it's being shown or hidden.
|
|
|
|
waterfall.on_hide();
|
2016-02-02 21:39:02 +00:00
|
|
|
update_modulation(modulation);
|
2016-02-02 21:42:00 +00:00
|
|
|
on_show_options_modulation();
|
2016-02-02 21:39:02 +00:00
|
|
|
waterfall.on_show();
|
2016-01-31 17:13:44 +00:00
|
|
|
}
|
|
|
|
|
2016-02-02 22:26:00 +00:00
|
|
|
void AnalogAudioView::remove_options_widget() {
|
|
|
|
if( options_widget ) {
|
|
|
|
remove_child(options_widget.get());
|
|
|
|
options_widget.reset();
|
|
|
|
}
|
2018-05-21 17:46:48 +00:00
|
|
|
|
2016-01-31 17:13:44 +00:00
|
|
|
field_lna.set_style(nullptr);
|
2016-02-02 21:42:00 +00:00
|
|
|
options_modulation.set_style(nullptr);
|
2016-02-02 22:26:00 +00:00
|
|
|
field_frequency.set_style(nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AnalogAudioView::set_options_widget(std::unique_ptr<Widget> new_widget) {
|
2016-04-22 17:01:17 +00:00
|
|
|
remove_options_widget();
|
|
|
|
|
2016-02-02 22:26:00 +00:00
|
|
|
if( new_widget ) {
|
|
|
|
options_widget = std::move(new_widget);
|
2016-05-25 18:39:03 +00:00
|
|
|
} else {
|
|
|
|
// TODO: Lame hack to hide options view due to my bad paint/damage algorithm.
|
|
|
|
options_widget = std::make_unique<Rectangle>(options_view_rect, style_options_group.background);
|
2016-02-02 22:26:00 +00:00
|
|
|
}
|
2016-05-25 18:39:03 +00:00
|
|
|
add_child(options_widget.get());
|
2016-02-02 22:26:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void AnalogAudioView::on_show_options_frequency() {
|
2016-02-16 23:17:06 +00:00
|
|
|
auto widget = std::make_unique<FrequencyOptionsView>(options_view_rect, &style_options_group);
|
2016-01-31 17:13:44 +00:00
|
|
|
|
2016-02-02 22:26:00 +00:00
|
|
|
widget->set_step(receiver_model.frequency_step());
|
|
|
|
widget->on_change_step = [this](rf::Frequency f) {
|
|
|
|
this->on_frequency_step_changed(f);
|
|
|
|
};
|
2016-02-10 23:34:33 +00:00
|
|
|
widget->set_reference_ppm_correction(persistent_memory::correction_ppb() / 1000);
|
2016-02-02 22:26:00 +00:00
|
|
|
widget->on_change_reference_ppm_correction = [this](int32_t v) {
|
|
|
|
this->on_reference_ppm_correction_changed(v);
|
|
|
|
};
|
|
|
|
|
|
|
|
set_options_widget(std::move(widget));
|
2016-04-22 17:16:42 +00:00
|
|
|
field_frequency.set_style(&style_options_group);
|
2016-01-31 17:13:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void AnalogAudioView::on_show_options_rf_gain() {
|
2016-02-16 23:17:06 +00:00
|
|
|
auto widget = std::make_unique<RadioGainOptionsView>(options_view_rect, &style_options_group);
|
2016-01-31 17:13:44 +00:00
|
|
|
|
2016-02-02 22:26:00 +00:00
|
|
|
set_options_widget(std::move(widget));
|
2016-04-22 17:16:42 +00:00
|
|
|
field_lna.set_style(&style_options_group);
|
2016-01-31 17:13:44 +00:00
|
|
|
}
|
|
|
|
|
2016-02-02 21:42:00 +00:00
|
|
|
void AnalogAudioView::on_show_options_modulation() {
|
2016-04-22 17:22:47 +00:00
|
|
|
std::unique_ptr<Widget> widget;
|
|
|
|
|
2016-02-02 21:42:00 +00:00
|
|
|
const auto modulation = static_cast<ReceiverModel::Mode>(receiver_model.modulation());
|
2016-04-22 17:22:47 +00:00
|
|
|
switch(modulation) {
|
|
|
|
case ReceiverModel::Mode::AMAudio:
|
|
|
|
widget = std::make_unique<AMOptionsView>(options_view_rect, &style_options_group);
|
2018-05-22 03:43:04 +00:00
|
|
|
waterfall.show_audio_spectrum_view(false);
|
|
|
|
text_ctcss.hidden(true);
|
2016-04-22 17:22:47 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case ReceiverModel::Mode::NarrowbandFMAudio:
|
2017-11-28 07:52:04 +00:00
|
|
|
widget = std::make_unique<NBFMOptionsView>(nbfm_view_rect, &style_options_group);
|
2018-05-22 03:43:04 +00:00
|
|
|
waterfall.show_audio_spectrum_view(false);
|
|
|
|
text_ctcss.hidden(false);
|
2016-04-22 17:22:47 +00:00
|
|
|
break;
|
2018-05-21 17:46:48 +00:00
|
|
|
|
|
|
|
case ReceiverModel::Mode::WidebandFMAudio:
|
2018-05-22 03:43:04 +00:00
|
|
|
waterfall.show_audio_spectrum_view(true);
|
|
|
|
text_ctcss.hidden(true);
|
2018-05-21 17:46:48 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case ReceiverModel::Mode::SpectrumAnalysis:
|
2018-05-22 03:43:04 +00:00
|
|
|
waterfall.show_audio_spectrum_view(false);
|
|
|
|
text_ctcss.hidden(true);
|
2018-05-21 17:46:48 +00:00
|
|
|
break;
|
|
|
|
|
2016-04-22 17:22:47 +00:00
|
|
|
default:
|
|
|
|
break;
|
2016-02-02 21:42:00 +00:00
|
|
|
}
|
2016-04-22 17:22:47 +00:00
|
|
|
|
|
|
|
set_options_widget(std::move(widget));
|
|
|
|
options_modulation.set_style(&style_options_group);
|
2016-02-02 21:42:00 +00:00
|
|
|
}
|
|
|
|
|
2016-01-31 17:13:44 +00:00
|
|
|
void AnalogAudioView::on_frequency_step_changed(rf::Frequency f) {
|
|
|
|
receiver_model.set_frequency_step(f);
|
|
|
|
field_frequency.set_step(f);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AnalogAudioView::on_reference_ppm_correction_changed(int32_t v) {
|
2016-02-10 23:34:33 +00:00
|
|
|
persistent_memory::set_correction_ppb(v * 1000);
|
2016-01-31 17:13:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void AnalogAudioView::on_headphone_volume_changed(int32_t v) {
|
2016-02-06 00:25:43 +00:00
|
|
|
const auto new_volume = volume_t::decibel(v - 99) + audio::headphone::volume_range().max;
|
2016-01-31 17:13:44 +00:00
|
|
|
receiver_model.set_headphone_volume(new_volume);
|
|
|
|
}
|
|
|
|
|
2016-02-02 21:39:02 +00:00
|
|
|
void AnalogAudioView::update_modulation(const ReceiverModel::Mode modulation) {
|
2016-02-06 00:25:43 +00:00
|
|
|
audio::output::mute();
|
2016-04-30 20:56:54 +00:00
|
|
|
record_view.stop();
|
2016-02-06 00:25:43 +00:00
|
|
|
|
2016-07-02 23:12:36 +00:00
|
|
|
baseband::shutdown();
|
|
|
|
|
2016-07-01 17:37:22 +00:00
|
|
|
portapack::spi_flash::image_tag_t image_tag;
|
|
|
|
switch(modulation) {
|
|
|
|
case ReceiverModel::Mode::AMAudio: image_tag = portapack::spi_flash::image_tag_am_audio; break;
|
|
|
|
case ReceiverModel::Mode::NarrowbandFMAudio: image_tag = portapack::spi_flash::image_tag_nfm_audio; break;
|
|
|
|
case ReceiverModel::Mode::WidebandFMAudio: image_tag = portapack::spi_flash::image_tag_wfm_audio; break;
|
|
|
|
case ReceiverModel::Mode::SpectrumAnalysis: image_tag = portapack::spi_flash::image_tag_wideband_spectrum; break;
|
|
|
|
default:
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
baseband::run_image(image_tag);
|
2017-02-01 08:53:26 +00:00
|
|
|
|
|
|
|
if (modulation == ReceiverModel::Mode::SpectrumAnalysis) {
|
|
|
|
baseband::set_spectrum(20000000, 127);
|
|
|
|
}
|
2016-07-01 17:37:22 +00:00
|
|
|
|
2016-02-02 21:39:02 +00:00
|
|
|
const auto is_wideband_spectrum_mode = (modulation == ReceiverModel::Mode::SpectrumAnalysis);
|
2016-07-27 21:51:37 +00:00
|
|
|
receiver_model.set_modulation(modulation);
|
2016-07-27 21:42:46 +00:00
|
|
|
receiver_model.set_sampling_rate(is_wideband_spectrum_mode ? 20000000 : 3072000);
|
2016-02-02 21:39:02 +00:00
|
|
|
receiver_model.set_baseband_bandwidth(is_wideband_spectrum_mode ? 12000000 : 1750000);
|
|
|
|
receiver_model.enable();
|
2016-02-05 23:22:28 +00:00
|
|
|
|
2016-04-30 18:25:04 +00:00
|
|
|
// TODO: This doesn't belong here! There's a better way.
|
|
|
|
size_t sampling_rate = 0;
|
|
|
|
switch(modulation) {
|
|
|
|
case ReceiverModel::Mode::AMAudio: sampling_rate = 12000; break;
|
|
|
|
case ReceiverModel::Mode::NarrowbandFMAudio: sampling_rate = 24000; break;
|
|
|
|
case ReceiverModel::Mode::WidebandFMAudio: sampling_rate = 48000; break;
|
|
|
|
default:
|
2016-04-30 20:56:54 +00:00
|
|
|
break;
|
2016-04-30 18:25:04 +00:00
|
|
|
}
|
2016-04-30 20:56:54 +00:00
|
|
|
record_view.set_sampling_rate(sampling_rate);
|
2016-04-30 18:25:04 +00:00
|
|
|
|
2016-04-30 20:56:54 +00:00
|
|
|
if( !is_wideband_spectrum_mode ) {
|
|
|
|
audio::output::unmute();
|
|
|
|
}
|
2016-04-30 18:25:04 +00:00
|
|
|
}
|
|
|
|
|
2018-04-19 19:50:32 +00:00
|
|
|
/*void AnalogAudioView::squelched() {
|
2017-05-18 10:06:11 +00:00
|
|
|
if (exit_on_squelch) nav_.pop();
|
2018-04-19 19:50:32 +00:00
|
|
|
}*/
|
2017-05-18 10:06:11 +00:00
|
|
|
|
2017-11-28 07:52:04 +00:00
|
|
|
void AnalogAudioView::handle_coded_squelch(const uint32_t value) {
|
|
|
|
float diff, min_diff = value;
|
|
|
|
size_t min_idx { 0 };
|
|
|
|
size_t c;
|
|
|
|
|
2018-05-21 17:46:48 +00:00
|
|
|
// Find nearest match
|
2017-12-06 13:20:51 +00:00
|
|
|
for (c = 0; c < tone_keys.size(); c++) {
|
2017-11-28 07:52:04 +00:00
|
|
|
diff = abs(((float)value / 100.0) - tone_keys[c].second);
|
|
|
|
if (diff < min_diff) {
|
|
|
|
min_idx = c;
|
|
|
|
min_diff = diff;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-05-21 17:46:48 +00:00
|
|
|
// Arbitrary confidence threshold
|
2017-11-28 07:52:04 +00:00
|
|
|
if (min_diff < 40)
|
|
|
|
text_ctcss.set("CTCSS " + tone_keys[min_idx].first);
|
|
|
|
else
|
|
|
|
text_ctcss.set("???");
|
|
|
|
}
|
|
|
|
|
2016-01-31 17:13:44 +00:00
|
|
|
} /* namespace ui */
|