2 |
use Moose::Role; |
use Moose::Role; |
3 |
|
|
4 |
with 'Frey::Session'; |
with 'Frey::Session'; |
5 |
|
with 'Frey::Class::Icon'; |
6 |
|
|
|
#use Continuity::Widget::DomNode; |
|
7 |
use Data::Dump qw/dump/; |
use Data::Dump qw/dump/; |
8 |
use Carp qw/confess cluck carp/; |
use Carp qw/confess cluck carp/; |
9 |
use File::Slurp; |
use File::Slurp; |
10 |
use Text::Tabs; # expand, unexpand |
use Text::Tabs; # expand, unexpand |
11 |
|
use Digest::MD5 qw/md5/; |
12 |
|
|
13 |
use lib 'lib'; |
use lib 'lib'; |
14 |
|
|
84 |
|
|
85 |
sub html_escape { |
sub html_escape { |
86 |
my ( $self, $html ) = @_; |
my ( $self, $html ) = @_; |
87 |
|
return '' unless defined $html; |
88 |
$html =~ s/($escape_re)/$escape{$1}/g; |
$html =~ s/($escape_re)/$escape{$1}/g; |
89 |
return $html; |
return $html; |
90 |
} |
} |
198 |
sub _add_css_js { |
sub _add_css_js { |
199 |
my ( $self, $what, $content ) = @_; |
my ( $self, $what, $content ) = @_; |
200 |
|
|
201 |
|
my $md5 = md5( $content ); |
202 |
|
return if $self->{_add_css_js_seen}->{$what}->{$md5}++; |
203 |
|
|
204 |
my $tag = $what eq 'css' ? 'style' : 'script'; |
my $tag = $what eq 'css' ? 'style' : 'script'; |
205 |
my $type = $what eq 'css' ? 'text/css' : 'text/javascript'; |
my $type = $what eq 'css' ? 'text/css' : 'text/javascript'; |
206 |
my $head; |
my $head; |
208 |
my ( $package, $path, $line ) = caller(1); |
my ( $package, $path, $line ) = caller(1); |
209 |
|
|
210 |
$content = "/$content" if $content !~ m{[\n\r]} && -e $content; |
$content = "/$content" if $content !~ m{[\n\r]} && -e $content; |
211 |
if ( $content =~ $re_html ) { |
if ( $content =~ $re_html && $what ne 'js' ) { |
212 |
$head = qq| |
$head = qq| |
213 |
$content |
$content |
214 |
<!-- $type via $package at $path line $line --> |
<!-- $type via $package at $path line $line --> |
250 |
our $reload_counter = 0; |
our $reload_counter = 0; |
251 |
|
|
252 |
|
|
253 |
=head2 page |
=head2 html_page |
254 |
|
|
255 |
$self->page( |
$self->html_page( |
256 |
title => 'page title', |
title => 'page title', |
257 |
head => '<!-- optional head markup -->', |
head => '<!-- optional head markup -->', |
258 |
body => '<b>Page Body</b>', |
body => '<b>Page Body</b>', |
263 |
our @status; |
our @status; |
264 |
sub status { @status }; |
sub status { @status }; |
265 |
|
|
266 |
our $icon_html; |
sub html_page { |
|
|
|
|
sub page { |
|
267 |
my $self = shift; |
my $self = shift; |
268 |
my $a = {@_}; |
my $a = {@_}; |
269 |
|
|
302 |
|
|
303 |
my $right = |
my $right = |
304 |
qq| |
qq| |
305 |
<span class="right"> |
<span class="frey-status-right"> |
306 |
<a title="reload $url" href="/reload$url">reload</a> |
<a title="reload $url" href="/reload$url">reload</a> |
307 |
<a title="$description" href="/exit$url" target="exit">$exit</a> |
<a title="$description" href="/exit$url" target="exit">$exit</a> |
308 |
</span> |
</span> |
313 |
my $revision = $svk->info->{Revision} || ''; |
my $revision = $svk->info->{Revision} || ''; |
314 |
$revision = $1 if $info->{'Mirrored From'} =~ m{Rev\.\s+(\d+)}; |
$revision = $1 if $info->{'Mirrored From'} =~ m{Rev\.\s+(\d+)}; |
315 |
|
|
316 |
$self->add_icon unless $icon_html; |
$self->add_icon; |
317 |
|
|
318 |
my $title = undef |
my $title = undef |
319 |
|| $a->{title} |
|| $a->{title} |
323 |
|
|
324 |
# $title =~ s{(\w)\w+::}{$1:}g; # XXX compress names of classes |
# $title =~ s{(\w)\w+::}{$1:}g; # XXX compress names of classes |
325 |
|
|
326 |
|
$self->add_css(qq| |
327 |
|
body { |
328 |
|
padding-bottom: 3em; /* don't overlap status line */ |
329 |
|
} |
330 |
|
|); |
331 |
|
|
332 |
my $html = join("\n", |
my $html = join("\n", |
333 |
qq|<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01//EN"><html><head>|, |
qq|<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01//EN"><html><head>|, |
334 |
$self->_head_html, |
$self->_head_html, |
335 |
qq|<title>$title</title>|, |
qq|<title>$title</title>|, |
336 |
'<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">', |
'<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">', |
337 |
( $icon_html || '<!-- no icon -->' ), |
( $self->icon_html ), |
338 |
( $a->{head} || '' ), |
( $a->{head} || '' ), |
339 |
qq| |
qq| |
340 |
</head><body> |
</head><body> |
485 |
|
|
486 |
sub clean_status { |
sub clean_status { |
487 |
my ($self) = shift; |
my ($self) = shift; |
488 |
|
warn "## clean_status"; |
489 |
@head = ( 'static/frey.css' ); |
@head = ( 'static/frey.css' ); |
490 |
@status = ( |
@status = ( |
491 |
{ 'ClassBrowser' => Frey::Class::Browser->new( usage_sort => 1, usage_on_top => 0 )->as_markup }, |
{ 'ClassBrowser' => Frey::Class::Browser->new( usage_sort => 1, usage_on_top => 0 )->as_markup }, |
492 |
{ 'Bookmarklets' => Frey::Bookmarklet->new->as_markup }, |
{ 'Bookmarklets' => Frey::Bookmarklet->new->as_markup }, |
493 |
{ 'INC' => Frey::INC->new->as_markup }, |
{ 'INC' => Frey::INC->new->as_markup }, |
494 |
); |
); |
|
$icon_html = ''; |
|
495 |
} |
} |
496 |
|
|
497 |
=head2 status_parts |
=head2 status_parts |
515 |
|
|
516 |
=cut |
=cut |
517 |
|
|
|
=head2 add_icon |
|
|
|
|
|
Frey::Foo->add_icon; # /static/icons/Frey/Foo.png |
|
|
Frey::Foo->add_icon('warning'); # /static/icons/Frey/Foo/warning.png |
|
|
|
|
|
=cut |
|
|
|
|
|
sub icon_path { |
|
|
my ($self,$class,$variant) = @_; |
|
|
|
|
|
sub icon_exists { |
|
|
my $class = shift; |
|
|
$class =~ s{::}{/}g; |
|
|
$class .= "/$variant" if $variant; |
|
|
my $icon_path = 'static/icons/' . $class . '.png'; |
|
|
return $icon_path if -e $icon_path; |
|
|
return; |
|
|
} |
|
|
|
|
|
my $path = icon_exists( $class ); |
|
|
if ( ! $path ) { |
|
|
my $super_class = $class; |
|
|
while ( $super_class =~ s{::[^:]+$}{} && ! $path ) { |
|
|
$path = icon_exists( $super_class ) unless $super_class eq 'Frey'; # don't default on Frey icon |
|
|
} |
|
|
} |
|
|
|
|
|
if ( ! $path ) { |
|
|
$self->TODO( "add icon for $class" . ( $variant ? " variant $variant" : '' ) ); |
|
|
return undef; |
|
|
} |
|
|
|
|
|
warn "# $class from $self icon_path $path" if $self->debug; |
|
|
return $path; |
|
|
} |
|
|
|
|
|
sub add_icon { |
|
|
my ($self,$variant) = @_; |
|
|
|
|
|
my $class = $self->class if $self->can('class'); |
|
|
#$class ||= $self->title; |
|
|
$class ||= ref($self); |
|
|
my $icon_path = $self->icon_path( $class, $variant ) || return; |
|
|
|
|
|
$icon_html .= qq|<link rel="icon" type="image/png" href="/$icon_path">|; |
|
|
warn "# using icon $icon_path"; |
|
|
|
|
|
=for later |
|
|
|
|
|
# FIXME http://en.wikipedia.org/wiki/Favicon suggest just rel="icon" but that doesn't seem to work! |
|
|
my $ico_path = $icon_path; |
|
|
$ico_path =~ s{png$}{ico}; |
|
|
if ( ! -e $ico_path ) { |
|
|
system "convert $icon_path $ico_path"; |
|
|
warn "# convert $icon_path $ico_path : $@"; |
|
|
} |
|
|
$icon_html .= qq|<link rel="shortcut icon" type="image/x-icon" href="/$ico_path">| if -e $ico_path; |
|
|
|
|
|
=cut |
|
|
|
|
|
} |
|
|
|
|
518 |
my $warn_colors = { |
my $warn_colors = { |
519 |
'#' => '#444', |
'#' => '#444', |
520 |
'##' => '#888', |
'##' => '#888', |