]>
iEval git - app-musicexpo.git/blob - lib/App/MusicExpo.pm
1 package App
::MusicExpo
;
6 our $VERSION = '1.000';
8 use Audio
::FLAC
::Header qw
//;
9 use HTML
::Template
::Compiled qw
//;
10 use Memoize qw
/memoize/;
11 use MP3
::Info qw
/get_mp3tag/;
12 use Ogg
::Vorbis
::Header
::PurePerl
;
13 use MP4
::Info qw
/get_mp4tag get_mp4info/;
16 use Encode qw
/encode/;
17 use File
::Basename qw
/fileparse/;
18 use Fcntl qw
/O_RDWR O_CREAT/;
20 use Storable qw
/thaw freeze/;
23 ##################################################
27 our $prefix='/music/';
32 'template:s' => \
$template,
33 'prefix:s' => \
$prefix,
39 my $flac=Audio
::FLAC
::Header
->new($file);
43 title
=> $flac->tags('TITLE'),
44 artist
=> $flac->tags('ARTIST'),
45 year
=> $flac->tags('DATE'),
46 album
=> $flac->tags('ALBUM'),
47 tracknumber
=> $flac->tags('TRACKNUMBER'),
48 tracktotal
=> $flac->tags('TRACKTOTAL'),
49 genre
=> $flac->tags('GENRE'),
50 file
=> scalar fileparse
$file,
56 my %tag = map { encode
'UTF-8', $_ } %{get_mp3tag
$file};
57 my @trkn = split m
#/#s, $tag{TRACKNUM} // '';
62 artist
=> $tag{ARTIST
},
65 tracknumber
=> $trkn[0],
66 tracktotal
=> $trkn[1],
68 file
=> scalar fileparse
$file,
74 my $ogg=Ogg
::Vorbis
::Header
::PurePerl
->new($file);
78 title
=> scalar $ogg->comment('TITLE'),
79 artist
=> scalar $ogg->comment('artist'),
80 year
=> scalar $ogg->comment('DATE'),
81 album
=> scalar $ogg->comment('ALBUM'),
82 tracknumber
=> scalar $ogg->comment('TRACKNUMBER'),
83 tracktotal
=> scalar $ogg->comment('TRACKTOTAL'),
84 genre
=> scalar $ogg->comment('GENRE'),
85 file
=> scalar fileparse
$file,
89 sub mp4_format
($){ ## no critic (ProhibitSubroutinePrototypes)
91 return 'AAC' if $encoding eq 'mp4a';
92 return 'ALAC' if $encoding eq 'alac';
98 my %tag = map { ref() ?
$_ : encode
'UTF-8', $_ } %{get_mp4tag
$file};
99 my %info = %{get_mp4info
$file};
102 format
=> mp4_format
$info{ENCODING
},
103 title
=> $tag{TITLE
},
104 artist
=> $tag{ARTIST
},
106 album
=> $tag{ALBUM
},
107 tracknumber
=> $tag{TRACKNUM
},
108 tracktotal
=> ($tag{TRKN
} ?
$tag{TRKN
}->[1] : undef),
109 genre
=> $tag{GENRE
},
110 file
=> scalar fileparse
$file,
115 '.flac' => \
&flacinfo
,
117 '.ogg' => \
&vorbisinfo
,
118 '.oga' => \
&vorbisinfo
,
125 "$_[0]|".(stat $_[0])[9]
128 sub make_fragment
{ join '-', map { lc =~ y/a-z0-9/_/csr } @_ }
132 tie
my %cache, 'DB_File', $cache, O_RDWR
|O_CREAT
, 0644; ## no critic (ProhibitTie)
133 $info{$_} = memoize
$info{$_}, INSTALL
=> undef, NORMALIZER
=> \
&normalizer
, LIST_CACHE
=> 'FAULT', SCALAR_CACHE
=> [HASH
=> \
%cache] for keys %info;
137 for my $file (@ARGV) {
138 my ($basename, undef, $suffix) = fileparse
$file, keys %info;
139 $files{$basename} //= [];
140 push @
{$files{$basename}}, thaw
scalar $info{$suffix}->($file);
143 my $ht=HTML
::Template
::Compiled
->new(
144 default_escape
=> 'HTML',
146 $template eq '' ?
(scalarref
=> \
$default_template) : (filename
=> $template),
150 for (sort keys %files) {
151 my @versions = @
{$files{$_}};
152 my %entry = (formats
=> [], map { $_ => '?' } qw
/title artist year album tracknumber tracktotal genre/);
153 for my $ver (@versions) {
154 push @
{$entry{formats
}}, {format
=> $ver->{format
}, file
=> $ver->{file
}};
155 for my $key (keys %$ver) {
156 $entry{$key} = $ver->{$key} if $ver->{$key} && $ver->{$key} ne '?';
159 delete $entry{$_} for qw
/format file/;
160 $entry{fragment
} = make_fragment
@entry{qw
/artist title/};
164 @files = sort { $a->{title
} cmp $b->{title
} } @files;
165 $ht->param(files
=> \
@files, prefix
=> $prefix);
166 print $ht->output; ## no critic (RequireCheckedSyscalls)
169 $default_template = <<'HTML';
172 <meta charset="utf-8">
173 <link rel="stylesheet" href="musicexpo.css">
174 <script async defer type="application/javascript" src="player.js"></script>
176 <div id="player"></div>
180 <tr><th>Title<th>Artist<th>Album<th>Genre<th>Track<th>Year<th>Type
181 <tbody><tmpl_loop files>
182 <tr><td class="title"><a href="#<tmpl_var fragment>" data-hash="#<tmpl_var fragment>"><tmpl_var title></a><td class="artist"><tmpl_var artist><td class="album"><tmpl_var album><td class="genre"><tmpl_var genre><td class="track"><tmpl_var tracknumber>/<tmpl_var tracktotal><td class="year"><tmpl_var year><td class="formats"><tmpl_loop formats><a href="<tmpl_var ...prefix><tmpl_var ESCAPE=URL file>"><tmpl_var format></a> </tmpl_loop></tmpl_loop>
194 App::MusicExpo - script which generates a HTML table of music tags
203 App::MusicExpo creates a HTML table from a list of songs.
205 The default template looks like:
207 | Title | Artist | Album | Genre | Track | Year | Type |
208 |---------+---------+-----------------+---------+-------+------+------|
209 | Cellule | Silence | L'autre endroit | Electro | 01/09 | 2005 | FLAC |
211 where the type is a download link. If you have multiple files with the same
212 basename (such as C<cellule.flac> and C<cellule.ogg>), they will be treated
213 as two versions of the same file, so a row will be created with two download
214 links, one for each format.
220 =item B<--template> I<template>
222 Path to the HTML::Template::Compiled template used for generating the music table. If '' (empty), uses the default format. Is empty by default.
224 =item B<--prefix> I<prefix>
226 Prefix for download links. Defaults to '/music/'.
228 =item B<--cache> I<filename>
230 Path to the cache file. Created if it does not exist. If '' (empty), disables caching. Is empty by default.
236 Marius Gavrilescu, E<lt>marius@ieval.roE<gt>
238 =head1 COPYRIGHT AND LICENSE
240 Copyright (C) 2013-2016 by Marius Gavrilescu
242 This library is free software; you can redistribute it and/or modify
243 it under the same terms as Perl itself, either Perl version 5.14.2 or,
244 at your option, any later version of Perl 5 you may have available.
This page took 0.065113 seconds and 5 git commands to generate.