123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370 |
- package Plugins::SqueezeESP32::Player;
- use strict;
- use base qw(Slim::Player::SqueezePlay);
- use Digest::MD5 qw(md5);
- use List::Util qw(min);
- use Slim::Utils::Log;
- use Slim::Utils::Prefs;
- use Plugins::SqueezeESP32::FirmwareHelper;
- my $sprefs = preferences('server');
- my $prefs = preferences('plugin.squeezeesp32');
- my $log = logger('plugin.squeezeesp32');
- {
- __PACKAGE__->mk_accessor('rw', qw(tone_update depth));
- }
- sub new {
- my $class = shift;
- my $client = $class->SUPER::new(@_);
- $client->init_accessor(
- tone_update => 0,
- );
- return $client;
- }
- our $defaultPrefs = {
- 'analogOutMode' => 0,
- 'bass' => 0,
- 'treble' => 0,
- 'lineInAlwaysOn' => 0,
- 'lineInLevel' => 50,
- 'menuItem' => [qw(
- NOW_PLAYING
- BROWSE_MUSIC
- RADIO
- PLUGIN_MY_APPS_MODULE_NAME
- PLUGIN_APP_GALLERY_MODULE_NAME
- FAVORITES
- GLOBAL_SEARCH
- PLUGIN_LINE_IN
- PLUGINS
- SETTINGS
- SQUEEZENETWORK_CONNECT
- )],
- };
- my $handlersAdded;
- sub model { 'squeezeesp32' }
- sub modelName { 'SqueezeESP32' }
- sub hasScrolling { 1 }
- sub hasIR { 1 }
- # TODO: add in settings when ready
- sub hasLineIn { 0 }
- sub hasHeadSubOut { 1 }
- sub maxTreble { 20 }
- sub minTreble { -13 }
- sub maxBass { 20 }
- sub minBass { -13 }
- sub init {
- my $client = shift;
- my ($id, $caps) = @_;
- my ($depth) = $caps =~ /Depth=(\d+)/;
- $client->depth($depth || 16);
- if (!$handlersAdded) {
- # Add a handler for line-in/out status changes
- Slim::Networking::Slimproto::addHandler( LIOS => \&lineInOutStatus );
- # Create a new event for sending LIOS updates
- Slim::Control::Request::addDispatch(
- ['lios', '_state'],
- [1, 0, 0, undef],
- );
- Slim::Control::Request::addDispatch(
- ['lios', 'linein', '_state'],
- [1, 0, 0, undef],
- );
- Slim::Control::Request::addDispatch(
- ['lios', 'lineout', '_state'],
- [1, 0, 0, undef],
- );
- $handlersAdded = 1;
- }
- $client->SUPER::init(@_);
- Plugins::SqueezeESP32::FirmwareHelper::init($client);
- main::INFOLOG && $log->is_info && $log->info("SqueezeESP player connected: " . $client->id);
- }
- sub initPrefs {
- my $client = shift;
- $sprefs->client($client)->init($defaultPrefs);
- $prefs->client($client)->init( {
- equalizer => [(0) x 10],
- artwork => undef,
- } );
- $prefs->setValidate({
- validator => sub {
- my ($pref, $new, $params, $old, $client) = @_;
- $new ||= [(0) x 10];
- foreach (0..9) {
- return 0 if $new->[$_] < $client->minBass;
- return 0 if $new->[$_] > $client->maxBass;
- }
- return 1;
- }
- }, 'equalizer');
- $client->SUPER::initPrefs;
- }
- sub power {
- my $client = shift;
- my $on = shift;
- my $res = $client->SUPER::power($on, @_);
- return $res unless defined $on;
- if ($on) {
- $client->update_artwork(1);
- } else {
- $client->clear_artwork(1);
- }
- return $res;
- }
- # Allow the player to define it's display width (and probably more)
- sub playerSettingsFrame {
- my $client = shift;
- my $data_ref = shift;
- my $value;
- my $id = unpack('C', $$data_ref);
- # New SETD command 0xfe for display width & height
- if ($id == 0xfe) {
- $value = (unpack('Cn', $$data_ref))[1];
- if ($value > 100 && $value < 400) {
- $prefs->client($client)->set('width', $value);
- my $height = (unpack('Cnn', $$data_ref))[2];
- $prefs->client($client)->set('height', $height || 0);
- $client->display->modes($client->display->build_modes);
- $client->display->widthOverride(1, $value);
- $client->update;
- main::INFOLOG && $log->is_info && $log->info("Setting player $value" . "x" . "$height for ", $client->name);
- }
- }
- $client->SUPER::playerSettingsFrame($data_ref);
- }
- sub bass {
- my ($client, $new) = @_;
- my $value = $client->SUPER::bass($new);
- $client->update_equalizer($value, [2, 1, 3]) if defined $new;
- return $value;
- }
- sub treble {
- my ($client, $new) = @_;
- my $value = $client->SUPER::treble($new);
- $client->update_equalizer($value, [8, 9, 7]) if defined $new;
- return $value;
- }
- sub send_equalizer {
- my ($client, $equalizer) = @_;
- $equalizer ||= $prefs->client($client)->get('equalizer') || [(0) x 10];
- my $size = @$equalizer;
- my $data = pack("c[$size]", @{$equalizer});
- $client->sendFrame( eqlz => \$data );
- }
- sub update_equalizer {
- my ($client, $value, $index) = @_;
- return if $client->tone_update;
- my $equalizer = $prefs->client($client)->get('equalizer');
- $equalizer->[$index->[0]] = $value;
- $equalizer->[$index->[1]] = int($value / 2 + 0.5);
- $equalizer->[$index->[2]] = int($value / 4 + 0.5);
- $prefs->client($client)->set('equalizer', $equalizer);
- }
- sub update_tones {
- my ($client, $equalizer) = @_;
- $client->tone_update(1);
- $sprefs->client($client)->set('bass', int(($equalizer->[1] * 2 + $equalizer->[2] + $equalizer->[3] * 4) / 7 + 0.5));
- $sprefs->client($client)->set('treble', int(($equalizer->[7] * 4 + $equalizer->[8] + $equalizer->[9] * 2) / 7 + 0.5));
- $client->tone_update(0);
- }
- sub update_artwork {
- my $client = shift;
- my $cprefs = $prefs->client($client);
- my $artwork = $cprefs->get('artwork') || return;
- return unless $artwork->{'enable'} && $client->display->isa("Plugins::SqueezeESP32::Graphics");
- my $header = pack('Nnn', $artwork->{'enable'}, $artwork->{'x'}, $artwork->{'y'});
- $client->sendFrame( grfa => \$header );
- $client->display->update;
- my $s = min($cprefs->get('height') - $artwork->{'y'}, $cprefs->get('width') - $artwork->{'x'});
- my $params = { force => shift || 0 };
- my $path = 'music/current/cover_' . $s . 'x' . $s . '_o.jpg';
- my $body = Slim::Web::Graphics::artworkRequest($client, $path, $params, \&send_artwork, undef, HTTP::Response->new);
- send_artwork($client, undef, \$body) if $body;
- }
- sub send_artwork {
- my ($client, $params, $dataref) = @_;
- # I'm not sure why we are called so often, so only send when needed
- my $md5 = md5($$dataref);
- return if $client->pluginData('artwork_md5') eq $md5 && !$params->{'force'};
- $client->pluginData('artwork', $dataref);
- $client->pluginData('artwork_md5', $md5);
- my $artwork = $prefs->client($client)->get('artwork') || {};
- my $length = length $$dataref;
- my $offset = 0;
- $log->info("got resized artwork (length: ", length $$dataref, ")");
- my $header = pack('Nnn', $length, $artwork->{'x'}, $artwork->{'y'});
- while ($length > 0) {
- $length = 1280 if $length > 1280;
- $log->info("sending grfa $length");
- my $data = $header . pack('N', $offset) . substr( $$dataref, 0, $length, '' );
- $client->sendFrame( grfa => \$data );
- $offset += $length;
- $length = length $$dataref;
- }
- }
- sub clear_artwork {
- my ($client, $force, $from) = @_;
- my $artwork = $prefs->client($client)->get('artwork');
- if ($artwork && $artwork->{'enable'}) {
- main::INFOLOG && $log->is_info && $log->info("artwork stop/clear " . ($from || ""));
- $client->pluginData('artwork_md5', '');
- # refresh screen and disable artwork when artwork was full screen (hack)
- if ((!$artwork->{'x'} && !$artwork->{'y'}) || $force) {
- $client->sendFrame(grfa => \("\x00"x4));
- $client->display->update;
- }
- }
- }
- sub config_artwork {
- my ($client) = @_;
- if ( my $artwork = $prefs->client($client)->get('artwork') ) {
- my $header = pack('Nnn', $artwork->{'enable'}, $artwork->{'x'}, $artwork->{'y'});
- $client->sendFrame( grfa => \$header );
- $client->display->update;
- }
- }
- sub reconnect {
- my $client = shift;
- $client->SUPER::reconnect(@_);
- $client->pluginData('artwork_md5', '');
- $client->config_artwork if $client->display->isa("Plugins::SqueezeESP32::Graphics");
- $client->send_equalizer;
- }
- # Change the analog output mode between headphone and sub-woofer
- # If no mode is specified, the value of the client's analogOutMode preference is used.
- # Otherwise the mode is temporarily changed to the given value without altering the preference.
- sub setAnalogOutMode {
- my $client = shift;
- # 0 = headphone (internal speakers off), 1 = sub out,
- # 2 = always on (internal speakers on), 3 = always off
- my $mode = shift;
- if (! defined $mode) {
- $mode = $sprefs->client($client)->get('analogOutMode');
- }
- my $data = pack('C', $mode);
- $client->sendFrame('audo', \$data);
- }
- # LINE_IN 0x01
- # LINE_OUT 0x02
- # HEADPHONE 0x04
- sub lineInConnected {
- my $state = Slim::Networking::Slimproto::voltage(shift) || return 0;
- return $state & 0x01 || 0;
- }
- sub lineOutConnected {
- my $state = Slim::Networking::Slimproto::voltage(shift) || return 0;
- return $state & 0x02 || 0;
- }
- sub lineInOutStatus {
- my ( $client, $data_ref ) = @_;
- my $state = unpack 'n', $$data_ref;
- my $oldState = {
- in => $client->lineInConnected(),
- out => $client->lineOutConnected(),
- };
- Slim::Networking::Slimproto::voltage( $client, $state );
- Slim::Control::Request::notifyFromArray( $client, [ 'lios', $state ] );
- if ($oldState->{in} != $client->lineInConnected()) {
- Slim::Control::Request::notifyFromArray( $client, [ 'lios', 'linein', $client->lineInConnected() ] );
- if ( Slim::Utils::PluginManager->isEnabled('Slim::Plugin::LineIn::Plugin')) {
- Slim::Plugin::LineIn::Plugin::lineInItem($client, 1);
- }
- }
- if ($oldState->{out} != $client->lineOutConnected()) {
- Slim::Control::Request::notifyFromArray( $client, [ 'lios', 'lineout', $client->lineOutConnected() ] );
- }
- }
- sub voltage {
- my $voltage = Slim::Networking::Slimproto::voltage(shift) || return 0;
- return sprintf("%.2f", ($voltage >> 4) / 128);
- }
- 1;
|