use Method::Signatures::Simple;
use FindBin;
use HTML::Zoom;
+use HTML::Zoom::FilterBuilder::Template;
+use List::Util qw(first);
{
package App::IdiotBox::Announcement;
+ sub id { shift->{id} }
sub made_at { shift->{made_at} }
sub bucket { shift->{bucket} }
sub video_count { shift->{video_count} }
sub slug { shift->{slug} }
sub name { shift->{name} }
- sub video_count { shift->{video_count} }
+ sub video_count {
+ exists $_[0]->{video_count}
+ ? $_[0]->{video_count}
+ : $_[0]->{videos}->count
+ }
+ sub videos { shift->{videos} }
+
+ package App::IdiotBox::Video;
+
+ sub slug { shift->{slug} }
+ sub name { shift->{name} }
+ sub author { shift->{author} }
+ sub details { shift->{details} }
+ sub bucket { shift->{bucket} }
+ sub file_name {
+ (my $s = join(' ', @{+shift}{qw(author name)})) =~ s/ /-/g;
+ $s;
+ }
+ sub url_path {
+ join('/', $_[0]->bucket->slug, $_[0]->slug);
+ }
}
default_config(
- template_dir => $FindBin::Bin.'/../share/html'
+ template_dir => 'share/html',
+ store => 'SQLite',
+ db_file => 'var/lib/idiotbox.db',
+ base_url => 'http://localhost:3000/',
+ base_dir => do { use FindBin; $FindBin::Bin },
);
+sub BUILD {
+ my $self = shift;
+ my $store;
+ ($store = $self->config->{store}) =~ /^(\w+)$/
+ or die "Store config should be just a name, got ${store} instead";
+ my $store_class = "App::IdiotBox::Store::${store}";
+ eval "require ${store_class}; 1"
+ or die "Couldn't load ${store} store: $@";
+ $store_class->bind($self);
+}
+
dispatch {
sub (/) { $self->show_front_page },
subdispatch sub (/*/...) {
sub (/) {
$self->show_bucket($bucket)
},
- sub (/*) {
+ sub (/*/) {
$self->show_video($bucket->videos->get({ slug => $_[1] }));
}
]
method recent_announcements { $self->{recent_announcements} }
+method buckets { $self->{buckets} }
+
method show_front_page {
my $ann = $self->recent_announcements;
$self->html_response(
- front_page => [
- '#announcement-list' => [
- -repeat_content => {
- repeat_for => $ann->map(sub { [
- '.fill-bucket-name' => [
- -replace_content => { replace_with => $_->bucket->name }
- ],
- '.fill-bucket-link' => [
- -set_attribute => { name => 'href', value => $_->bucket->slug.'/' }
- ],
- '.fill-new-videos' => [
- -replace_content => { replace_with => $_->video_count }
- ],
- '.fill-total-videos' => [
- -replace_content => { replace_with => $_->bucket->video_count }
- ],
- ] })->as_stream
- }
- ]
- ]
+ front_page => sub {
+ $_->select('#announcement-list')
+ ->repeat_content($ann->map(sub {
+ my $obj = $_;
+ sub {
+ $_->select('.bucket-name')->replace_content($obj->bucket->name)
+ ->select('.made-at')->replace_content($obj->made_at)
+ ->select('.bucket-link')->set_attribute({
+ name => 'href', value => $obj->bucket->slug.'/'
+ })
+ ->select('.new-videos')->replace_content($obj->video_count)
+ ->select('.total-videos')->replace_content(
+ $obj->bucket->video_count
+ )
+ }
+ }))
+ }
);
}
method show_bucket ($bucket) {
- $self->html_response(bucket => [
- '.fill-bucket-name' => [
- -replace_content => { replace_with => $bucket->name }
- ],
- '#video-list' => [
- -repeat_content => {
- repeat_for => $bucket->videos->map(sub { [
- '.fill-video-name' => [
- -replace_content => { replace_with => $_->name }
- ],
- '.fill-video-author' => [
- -replace_content => { replace_with => $_->author }
- ],
- '.fill-video-link' => [
- -set_attribute => {
- name => 'href', value => $_->slug.'/'
- }
- ],
- ] })->as_stream
- }
- ]
- ]);
+ $self->html_response(bucket => sub {
+ $_->select('.bucket-name')->replace_content($bucket->name)
+ ->select('#video-list')->repeat_content($bucket->videos->map(sub {
+ my $video = $_;
+ sub {
+ $_->select('.video-name')->replace_content($video->name)
+ ->select('.video-author')->replace_content($video->author)
+ ->select('.video-link')->set_attribute(
+ { name => 'href', value => $video->slug.'/' }
+ )
+ }
+ }))
+ });
}
method show_video ($video) {
- $self->html_response(video => [
- '.fill-video-name' => [
- -replace_content => { replace_with => $video->name }
- ],
- '.fill-author-name' => [
- -replace_content => { replace_with => $video->author }
- ],
- '.fill-bucket-link' => [
- -set_attribute => { name => 'href', value => '../' }
- ],
- '.fill-bucket-name' => [
- -replace_content => { replace_with => $video->bucket->name }
- ],
- '.fill-video-details' => [
- -replace_content => { replace_with => $video->details }
- ]
- ]);
+ my $video_file = first {
+ -e join('/', $self->config->{base_dir}, $_)
+ } map {
+ join('/', $video->bucket->slug, $video->slug, $video->file_name.".$_")
+ } qw(flv m4v);
+ $self->html_response(video => sub {
+ my $video_url =
+ $self->base_url
+ .$video_file;
+
+ $_->select('.video-name')->replace_content($video->name)
+ ->select('.author-name')->replace_content($video->author)
+ ->select('.bucket-link')->set_attribute(
+ { name => 'href', value => '../' }
+ )
+ ->select('.bucket-name')->replace_content($video->bucket->name)
+ ->select('.video-details')->replace_content($video->details)
+ ->select('script')->template_text_raw({ video_url => $video_url });
+ });
}
method html_response ($template_name, $selectors) {
- my $io = $self->_zoom_for($template_name => $selectors)->as_readable_fh;
+ my $io = $self->_zoom_for($template_name => $selectors)->to_fh;
return [ 200, [ 'Content-Type' => 'text/html' ], $io ]
}
HTML::Zoom->from_file(
$self->_template_filename_for($template_name)
)
- ->with_selectors(
- '#main-content' => [
- -capture_events => { into => \@body }
- ]
- )
+ ->select('#main-content')->collect_content({ into => \@body })
->run;
- $self->_layout_zoom->with_selectors(
- '#main-content' => [
- -replace_content_events => { replace_with => \@body }
- ]
- )->to_zoom;
- })->with_selectors($selectors)
+ $self->_layout_zoom
+ ->select('#main-content')->replace_content(\@body)
+ ->memoize;
+ })->apply($selectors);
+}
+
+method base_url {
+ $self->{base_url} ||= do {
+ (my $u = $self->config->{base_url}) =~ s/\/$//;
+ "${u}/";
+ }
+}
+
+method _run_cli {
+ unless (@ARGV == 1 && $ARGV[0] eq 'import') {
+ return $self->SUPER::_run_cli(@_);
+ }
+ $self->cli_import;
+}
+
+method _cli_usage {
+ "To import data into your idiotbox install, chdir into a directory\n".
+ "containing video files and run:\n".
+ "\n".
+ " $0 import\n".
+ "\n".
+ $self->SUPER::_cli_usage(@_);
+}
+
+method cli_import {
+ require App::IdiotBox::Importer;
+ App::IdiotBox::Importer->run($self);
}
1;