Introducing a very barebones admin UI for managing buckets. I would
[catagits/App-IdiotBox.git] / lib / App / IdiotBox.pm
index c91d6c8..a915049 100644 (file)
@@ -1,5 +1,6 @@
 package App::IdiotBox;
 
+use App::IdiotBox::Common qw(@SupportedFormats);
 use Web::Simple __PACKAGE__;
 use FindBin;
 use HTML::Zoom;
@@ -41,13 +42,15 @@ use List::Util qw(first);
   }
 }
 
-default_config(
+has $_ => (is => 'ro') for qw(recent_announcements buckets);
+
+sub default_config {
   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;
@@ -60,24 +63,68 @@ sub BUILD {
   $store_class->bind($self);
 }
   
-dispatch {
+sub dispatch_request {
+  my $self = shift;
   sub (/) { $self->show_front_page },
-  subdispatch sub (/*/...) {
+  sub (/admin/) {
+    sub (%new_name=&new_slug=) {
+        my ($self, $name, $slug) = @_;
+
+       unless ($name && $slug) {
+               return $self->show_admin_page(error => "Please enter a name and a bucket");
+       }
+       if ($name =~ /^\s+$/ || $slug =~ /^\s+$/) {
+               return $self->show_admin_page(error => "Names/buckets must not be all whitespace");
+       }
+
+        $slug =~ s/ /-/g;
+
+        my $nb = $self->buckets->add(bless({
+          slug => $slug,
+          name => $name,
+        }, 'App::IdiotBox::Bucket'));
+
+        $self->show_admin_page;
+    },
+  },
+  sub (/admin/) { $self->show_admin_page },
+
+  sub (/admin/*/...) {
     my $bucket = $self->buckets->get({ slug => $_[1] });
-    [
-      sub (/) {
-        $self->show_bucket($bucket)
-      },
-      sub (/*/) {
-        $self->show_video($bucket->videos->get({ slug => $_[1] }));
+    sub (%new_name=) {
+      my ($self, $new_name) = @_;
+      if (!$new_name) {
+        return $self->show_edit_bucket_page($bucket, error => "Please enter a new name");
+      } elsif ($new_name =~ /^\s+$/) {
+        return $self->show_edit_bucket_page($bucket, error => "Names must not be all whitespace");
       }
-    ]
+      $self->buckets->replace($bucket, bless({
+          slug => $bucket->slug,
+          name => $new_name,
+        }, 'App::IdiotBox::Bucket'));
+      $self->show_bucket_edited_page($bucket);
+    },
+    sub (/) {
+      $self->show_edit_bucket_page($bucket);
+    },
+    sub (/delete/) {
+      $self->show_confirm_delete_bucket_page($bucket)
+    },
+    sub (/delete/yes/) {
+      $self->buckets->remove({ slug => $bucket->slug });
+      $self->show_bucket_deleted_page($bucket->slug);
+    },
+  },
+  sub (/*/...) {
+    my $bucket = $self->buckets->get({ slug => $_[1] });
+    sub (/) {
+      $self->show_bucket($bucket)
+    },
+    sub (/*/) {
+      $self->show_video($bucket->videos->get({ slug => $_[1] }));
+    }
   }
-};
-
-sub recent_announcements { shift->{recent_announcements} }
-
-sub buckets { shift->{buckets} }
+}
 
 sub show_front_page {
   my $self = shift;
@@ -90,9 +137,9 @@ sub show_front_page {
             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('.bucket-link')->set_attribute(
+                    'href' => $obj->bucket->slug.'/'
+                  )
                 ->select('.new-videos')->replace_content($obj->video_count)
                 ->select('.total-videos')->replace_content(
                     $obj->bucket->video_count
@@ -103,6 +150,68 @@ sub show_front_page {
   );
 }
 
+sub show_admin_page {
+  my $self = shift;
+  my %opts = @_;
+  my $error = $opts{error} || '';
+
+  my $bucket = $self->buckets;
+  $self->html_response(
+    admin => sub {
+      $_->select('#bucket-list')
+        ->repeat_content($bucket->map(sub {
+            my $obj = $_;
+            sub {
+              $_->select('.bucket-slug')->replace_content($obj->slug)
+                ->select('.bucket-name')->replace_content($obj->name)
+                ->select('.edit-link')->set_attribute(
+                    'href' => $obj->slug.'/'
+                  )
+                ->select('.delete-link')->set_attribute(
+                    'href' => $obj->slug.'/delete/'
+                  )
+            }
+          }))
+        ->select('.error-text')->replace_content($error)
+
+    }
+
+  );
+}
+
+sub show_confirm_delete_bucket_page {
+  my ($self, $bucket) = @_;
+  $self->html_response('delete' => sub {
+    $_->select('.bucket-name')->replace_content($bucket->name)
+      ->select('.confirm-yes')->set_attribute(
+        'href' => 'yes/'
+        )
+  });
+}
+
+sub show_edit_bucket_page {
+  my ($self, $bucket, %opt) = @_;
+  my $error = $opt{error} || '';
+  $self->html_response('edit' => sub {
+    $_->select('.bucket-name')->replace_content($bucket->name)
+      ->select('.error-text')->replace_content($error);
+  });
+}
+
+sub show_bucket_deleted_page {
+  my ($self, $name) = @_;
+  $self->html_response('deleted' => sub {
+    $_->select('.bucket-name')->replace_content($name)
+  });
+}
+
+sub show_bucket_edited_page {
+  my ($self, $name) = @_;
+  $self->html_response('edited' => sub {
+    $_->select('.bucket-name')->replace_content($name)
+  });
+}
+
 sub show_bucket {
   my ($self, $bucket) = @_;
   $self->html_response(bucket => sub {
@@ -113,7 +222,7 @@ sub show_bucket {
             $_->select('.video-name')->replace_content($video->name)
               ->select('.video-author')->replace_content($video->author)
               ->select('.video-link')->set_attribute(
-                  { name => 'href', value => $video->slug.'/' }
+                  href => $video->slug.'/'
                 )
           }
         }))
@@ -126,7 +235,7 @@ sub show_video {
     -e join('/', $self->config->{base_dir}, $_)
   } map {
     join('/', $video->bucket->slug, $video->slug, $video->file_name.".$_")
-  } qw(flv m4v);
+  } @SupportedFormats;
   $self->html_response(video => sub {
     my $video_url = 
       $self->base_url
@@ -135,7 +244,7 @@ sub show_video {
     $_->select('.video-name')->replace_content($video->name)
       ->select('.author-name')->replace_content($video->author)
       ->select('.bucket-link')->set_attribute(
-          { name => 'href', value => '../' }
+          href => '../'
         )
       ->select('.bucket-name')->replace_content($video->bucket->name)
       ->select('.video-details')->replace_content($video->details)