You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
243 lines
7.8 KiB
243 lines
7.8 KiB
/*
|
|
* Copyright 2023 jacqueline <me@jacqueline.id.au>
|
|
*
|
|
* SPDX-License-Identifier: GPL-3.0-only
|
|
*/
|
|
|
|
#include "screen_playing.hpp"
|
|
#include <sys/_stdint.h>
|
|
#include <memory>
|
|
|
|
#include "core/lv_obj.h"
|
|
#include "core/lv_obj_tree.h"
|
|
#include "database.hpp"
|
|
#include "esp_log.h"
|
|
#include "extra/layouts/flex/lv_flex.h"
|
|
#include "extra/layouts/grid/lv_grid.h"
|
|
#include "font/lv_symbol_def.h"
|
|
#include "future_fetcher.hpp"
|
|
#include "lvgl.h"
|
|
|
|
#include "core/lv_group.h"
|
|
#include "core/lv_obj_pos.h"
|
|
#include "event_queue.hpp"
|
|
#include "extra/widgets/list/lv_list.h"
|
|
#include "extra/widgets/menu/lv_menu.h"
|
|
#include "extra/widgets/spinner/lv_spinner.h"
|
|
#include "future_fetcher.hpp"
|
|
#include "hal/lv_hal_disp.h"
|
|
#include "index.hpp"
|
|
#include "misc/lv_anim.h"
|
|
#include "misc/lv_area.h"
|
|
#include "misc/lv_color.h"
|
|
#include "track.hpp"
|
|
#include "ui_events.hpp"
|
|
#include "ui_fsm.hpp"
|
|
#include "widgets/lv_bar.h"
|
|
#include "widgets/lv_btn.h"
|
|
#include "widgets/lv_img.h"
|
|
#include "widgets/lv_label.h"
|
|
|
|
namespace ui {
|
|
namespace screens {
|
|
|
|
static constexpr std::size_t kMaxUpcoming = 10;
|
|
|
|
static lv_style_t scrubber_style;
|
|
|
|
auto info_label(lv_obj_t* parent) -> lv_obj_t* {
|
|
lv_obj_t* label = lv_label_create(parent);
|
|
lv_obj_set_size(label, lv_pct(100), LV_SIZE_CONTENT);
|
|
lv_label_set_long_mode(label, LV_LABEL_LONG_DOT);
|
|
lv_obj_center(label);
|
|
return label;
|
|
}
|
|
|
|
auto control_button(lv_obj_t* parent, char* icon) -> lv_obj_t* {
|
|
lv_obj_t* button = lv_img_create(parent);
|
|
lv_obj_set_size(button, LV_SIZE_CONTENT, LV_SIZE_CONTENT);
|
|
lv_img_set_src(button, icon);
|
|
return button;
|
|
}
|
|
|
|
auto next_up_label(lv_obj_t* parent, const std::string& text) -> lv_obj_t* {
|
|
lv_obj_t* label = lv_label_create(parent);
|
|
lv_label_set_text(label, text.c_str());
|
|
lv_obj_set_size(label, lv_pct(100), LV_SIZE_CONTENT);
|
|
lv_label_set_long_mode(label, LV_LABEL_LONG_SCROLL_CIRCULAR);
|
|
return label;
|
|
}
|
|
|
|
Playing::Playing(std::weak_ptr<database::Database> db, audio::TrackQueue* queue)
|
|
: db_(db),
|
|
queue_(queue),
|
|
track_(),
|
|
next_tracks_(),
|
|
new_track_(),
|
|
new_next_tracks_() {
|
|
lv_obj_set_layout(root_, LV_LAYOUT_FLEX);
|
|
lv_obj_set_size(root_, lv_pct(100), lv_pct(200));
|
|
lv_obj_set_flex_flow(root_, LV_FLEX_FLOW_COLUMN);
|
|
lv_obj_set_flex_align(root_, LV_FLEX_ALIGN_CENTER, LV_FLEX_ALIGN_START,
|
|
LV_FLEX_ALIGN_START);
|
|
|
|
lv_obj_t* above_fold_container = lv_obj_create(root_);
|
|
lv_obj_set_layout(above_fold_container, LV_LAYOUT_FLEX);
|
|
lv_obj_set_size(above_fold_container, lv_pct(100), lv_disp_get_ver_res(NULL));
|
|
lv_obj_set_flex_flow(above_fold_container, LV_FLEX_FLOW_COLUMN);
|
|
lv_obj_set_flex_align(above_fold_container, LV_FLEX_ALIGN_CENTER,
|
|
LV_FLEX_ALIGN_CENTER, LV_FLEX_ALIGN_END);
|
|
|
|
lv_obj_t* info_container = lv_obj_create(above_fold_container);
|
|
lv_obj_set_layout(info_container, LV_LAYOUT_FLEX);
|
|
lv_obj_set_size(info_container, lv_pct(100), 80);
|
|
lv_obj_set_flex_flow(info_container, LV_FLEX_FLOW_COLUMN);
|
|
lv_obj_set_flex_align(info_container, LV_FLEX_ALIGN_CENTER,
|
|
LV_FLEX_ALIGN_CENTER, LV_FLEX_ALIGN_END);
|
|
|
|
artist_label_ = info_label(info_container);
|
|
album_label_ = info_label(info_container);
|
|
title_label_ = info_label(info_container);
|
|
|
|
scrubber_ = lv_bar_create(above_fold_container);
|
|
lv_obj_set_size(scrubber_, lv_pct(100), 5);
|
|
lv_bar_set_range(scrubber_, 0, 100);
|
|
lv_bar_set_value(scrubber_, 0, LV_ANIM_OFF);
|
|
|
|
lv_style_init(&scrubber_style);
|
|
lv_style_set_bg_color(&scrubber_style, lv_color_black());
|
|
lv_obj_add_style(scrubber_, &scrubber_style, LV_PART_INDICATOR);
|
|
|
|
lv_obj_t* controls_container = lv_obj_create(above_fold_container);
|
|
lv_obj_set_size(controls_container, lv_pct(100), 20);
|
|
lv_obj_set_flex_flow(controls_container, LV_FLEX_FLOW_ROW);
|
|
lv_obj_set_flex_align(controls_container, LV_FLEX_ALIGN_SPACE_EVENLY,
|
|
LV_FLEX_ALIGN_CENTER, LV_FLEX_ALIGN_CENTER);
|
|
|
|
play_pause_control_ = control_button(controls_container, LV_SYMBOL_PLAY);
|
|
lv_group_add_obj(group_, play_pause_control_);
|
|
lv_group_add_obj(group_, control_button(controls_container, LV_SYMBOL_PREV));
|
|
lv_group_add_obj(group_, control_button(controls_container, LV_SYMBOL_NEXT));
|
|
lv_group_add_obj(group_,
|
|
control_button(controls_container, LV_SYMBOL_SHUFFLE));
|
|
lv_group_add_obj(group_, control_button(controls_container, LV_SYMBOL_LOOP));
|
|
|
|
lv_obj_t* next_up_header = lv_obj_create(above_fold_container);
|
|
lv_obj_set_size(next_up_header, lv_pct(100), 15);
|
|
lv_obj_set_flex_flow(next_up_header, LV_FLEX_FLOW_ROW);
|
|
lv_obj_set_flex_align(next_up_header, LV_FLEX_ALIGN_START, LV_FLEX_ALIGN_END,
|
|
LV_FLEX_ALIGN_END);
|
|
|
|
lv_obj_t* next_up_title = lv_label_create(next_up_header);
|
|
lv_label_set_text(next_up_title, "Next up...");
|
|
lv_obj_set_height(next_up_title, lv_pct(100));
|
|
lv_obj_set_flex_grow(next_up_title, 1);
|
|
|
|
lv_obj_t* next_up_hint = lv_label_create(next_up_header);
|
|
lv_label_set_text(next_up_hint, LV_SYMBOL_DOWN);
|
|
lv_obj_set_size(next_up_hint, LV_SIZE_CONTENT, lv_pct(100));
|
|
|
|
next_up_container_ = lv_obj_create(root_);
|
|
lv_obj_set_layout(next_up_container_, LV_LAYOUT_FLEX);
|
|
lv_obj_set_width(next_up_container_, lv_pct(100));
|
|
lv_obj_set_flex_grow(next_up_container_, 1);
|
|
lv_obj_set_flex_flow(next_up_container_, LV_FLEX_FLOW_COLUMN);
|
|
lv_obj_set_flex_align(next_up_container_, LV_FLEX_ALIGN_CENTER,
|
|
LV_FLEX_ALIGN_CENTER, LV_FLEX_ALIGN_END);
|
|
|
|
OnTrackUpdate();
|
|
OnQueueUpdate();
|
|
}
|
|
|
|
Playing::~Playing() {}
|
|
|
|
auto Playing::OnTrackUpdate() -> void {
|
|
auto current = queue_->GetCurrent();
|
|
if (!current) {
|
|
return;
|
|
}
|
|
if (track_ && track_->data().id() == *current) {
|
|
return;
|
|
}
|
|
auto db = db_.lock();
|
|
if (!db) {
|
|
return;
|
|
}
|
|
new_track_.reset(new database::FutureFetcher<std::optional<database::Track>>(
|
|
db->GetTrack(*current)));
|
|
}
|
|
|
|
auto Playing::OnPlaybackUpdate(uint32_t pos_seconds, uint32_t new_duration)
|
|
-> void {
|
|
if (!track_) {
|
|
return;
|
|
}
|
|
lv_bar_set_range(scrubber_, 0, new_duration);
|
|
lv_bar_set_value(scrubber_, pos_seconds, LV_ANIM_ON);
|
|
}
|
|
|
|
auto Playing::OnQueueUpdate() -> void {
|
|
auto current = queue_->GetUpcoming(kMaxUpcoming);
|
|
auto db = db_.lock();
|
|
if (!db) {
|
|
return;
|
|
}
|
|
new_next_tracks_.reset(
|
|
new database::FutureFetcher<std::vector<std::optional<database::Track>>>(
|
|
db->GetBulkTracks(current)));
|
|
}
|
|
|
|
auto Playing::Tick() -> void {
|
|
if (new_track_ && new_track_->Finished()) {
|
|
auto res = new_track_->Result();
|
|
new_track_.reset();
|
|
if (res && *res) {
|
|
BindTrack(**res);
|
|
}
|
|
}
|
|
if (new_next_tracks_ && new_next_tracks_->Finished()) {
|
|
auto res = new_next_tracks_->Result();
|
|
new_next_tracks_.reset();
|
|
if (res) {
|
|
std::vector<database::Track> filtered;
|
|
for (const auto& t : *res) {
|
|
if (t) {
|
|
filtered.push_back(*t);
|
|
}
|
|
}
|
|
ApplyNextUp(filtered);
|
|
}
|
|
}
|
|
}
|
|
|
|
auto Playing::BindTrack(const database::Track& t) -> void {
|
|
track_ = t;
|
|
|
|
lv_label_set_text(artist_label_,
|
|
t.tags().at(database::Tag::kArtist).value_or("").c_str());
|
|
lv_label_set_text(album_label_,
|
|
t.tags().at(database::Tag::kAlbum).value_or("").c_str());
|
|
lv_label_set_text(title_label_, t.TitleOrFilename().c_str());
|
|
|
|
std::optional<int> duration = t.tags().duration;
|
|
lv_bar_set_range(scrubber_, 0, duration.value_or(1));
|
|
lv_bar_set_value(scrubber_, 0, LV_ANIM_OFF);
|
|
}
|
|
|
|
auto Playing::ApplyNextUp(const std::vector<database::Track>& tracks) -> void {
|
|
// TODO(jacqueline): Do a proper diff to maintain selection.
|
|
int children = lv_obj_get_child_cnt(next_up_container_);
|
|
while (children > 0) {
|
|
lv_obj_del(lv_obj_get_child(next_up_container_, 0));
|
|
children--;
|
|
}
|
|
|
|
next_tracks_ = tracks;
|
|
for (const auto& track : next_tracks_) {
|
|
lv_group_add_obj(
|
|
group_, next_up_label(next_up_container_, track.TitleOrFilename()));
|
|
}
|
|
}
|
|
|
|
} // namespace screens
|
|
} // namespace ui
|
|
|