line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
# mamgal - a program for creating static image galleries |
2
|
|
|
|
|
|
|
# Copyright 2007-2010 Marcin Owsiany |
3
|
|
|
|
|
|
|
# See the README file for license information |
4
|
|
|
|
|
|
|
# Any interesting entry (picture or subdirectory) |
5
|
|
|
|
|
|
|
package App::MaMGal::Entry; |
6
|
5
|
|
|
5
|
|
33413
|
use strict; |
|
5
|
|
|
|
|
10
|
|
|
5
|
|
|
|
|
157
|
|
7
|
5
|
|
|
5
|
|
27
|
use warnings; |
|
5
|
|
|
|
|
7
|
|
|
5
|
|
|
|
|
156
|
|
8
|
5
|
|
|
5
|
|
23
|
use base 'App::MaMGal::Base'; |
|
5
|
|
|
|
|
7
|
|
|
5
|
|
|
|
|
1778
|
|
9
|
5
|
|
|
5
|
|
26
|
use Carp; |
|
5
|
|
|
|
|
8
|
|
|
5
|
|
|
|
|
275
|
|
10
|
5
|
|
|
5
|
|
25
|
use File::Basename; |
|
5
|
|
|
|
|
8
|
|
|
5
|
|
|
|
|
376
|
|
11
|
5
|
|
|
5
|
|
26
|
use File::stat; |
|
5
|
|
|
|
|
8
|
|
|
5
|
|
|
|
|
44
|
|
12
|
|
|
|
|
|
|
|
13
|
|
|
|
|
|
|
sub init |
14
|
|
|
|
|
|
|
{ |
15
|
90
|
|
|
90
|
0
|
120
|
my $self = shift; |
16
|
90
|
100
|
|
|
|
386
|
my $dirname = shift or croak "Need dir"; # the directory which contains this entry, relative to WD or absolute |
17
|
87
|
100
|
|
|
|
221
|
my $basename = shift or croak "Need basename"; # under $dirname |
18
|
84
|
100
|
33
|
|
|
369
|
confess "A basename of \".\" used when other would be possible (last component of $dirname)" if $basename eq '.' and not ($dirname eq '.' or $dirname eq '/'); |
|
|
|
66
|
|
|
|
|
19
|
81
|
100
|
|
|
|
261
|
confess "Basename [$basename] contains a slash" if $basename =~ m{/}; |
20
|
|
|
|
|
|
|
# We might not be able to get stat information (e.g. no execute permission on containing directory), so do not croak |
21
|
78
|
|
|
|
|
109
|
my $stat = shift; |
22
|
78
|
100
|
66
|
|
|
540
|
confess "Third argument must be a File::stat, if provided" unless (not defined $stat) or (ref $stat and $stat->isa('File::stat')); |
|
|
|
66
|
|
|
|
|
23
|
75
|
100
|
|
|
|
879
|
confess "At most 3 args expected, got fourth: [$_[0]]" if @_; |
24
|
|
|
|
|
|
|
|
25
|
72
|
|
|
|
|
188
|
$self->{dir_name} = $dirname; |
26
|
72
|
|
|
|
|
140
|
$self->{base_name} = $basename; |
27
|
72
|
|
|
|
|
105
|
$self->{stat} = $stat; |
28
|
72
|
|
|
|
|
166
|
$self->{path_name} = $dirname.'/'.$basename; |
29
|
72
|
|
|
|
|
300
|
$self->{tools} = {}; |
30
|
|
|
|
|
|
|
} |
31
|
|
|
|
|
|
|
|
32
|
|
|
|
|
|
|
sub add_tools |
33
|
|
|
|
|
|
|
{ |
34
|
72
|
|
|
72
|
0
|
119
|
my $self = shift; |
35
|
72
|
|
|
|
|
88
|
my $tools = shift; |
36
|
72
|
|
|
|
|
242
|
foreach (keys %$tools) { $self->{tools}->{$_} = $tools->{$_} } |
|
144
|
|
|
|
|
457
|
|
37
|
|
|
|
|
|
|
} |
38
|
|
|
|
|
|
|
|
39
|
|
|
|
|
|
|
sub tools |
40
|
|
|
|
|
|
|
{ |
41
|
156
|
|
|
156
|
0
|
468
|
my $self = shift; |
42
|
156
|
|
|
|
|
1085
|
return $self->{tools}; |
43
|
|
|
|
|
|
|
} |
44
|
|
|
|
|
|
|
|
45
|
|
|
|
|
|
|
sub logger |
46
|
|
|
|
|
|
|
{ |
47
|
6
|
|
|
6
|
0
|
3223
|
my $self = shift; |
48
|
6
|
|
|
|
|
20
|
return $self->tools->{logger}; |
49
|
|
|
|
|
|
|
} |
50
|
|
|
|
|
|
|
|
51
|
|
|
|
|
|
|
# TODO: element should not have a need to know its index, container should be able to tell it simply given the object |
52
|
0
|
|
|
0
|
0
|
0
|
sub element_index { $_[0]->{element_index} } |
53
|
0
|
|
|
0
|
0
|
0
|
sub set_element_index { $_[0]->{element_index} = $_[1] } |
54
|
6
|
|
|
6
|
0
|
9474
|
sub name { $_[0]->{base_name} } |
55
|
6
|
|
|
6
|
0
|
3803
|
sub description { '' } |
56
|
6
|
|
|
6
|
0
|
505
|
sub set_container { $_[0]->{container} = $_[1] } |
57
|
|
|
|
|
|
|
|
58
|
|
|
|
|
|
|
sub container |
59
|
|
|
|
|
|
|
{ |
60
|
6
|
|
|
6
|
0
|
4012
|
my $self = shift; |
61
|
6
|
50
|
|
|
|
56
|
unless (defined $self->{container}) { |
62
|
|
|
|
|
|
|
# TODO this will lead to creation of a strange split tree if it |
63
|
|
|
|
|
|
|
# is traversed again from container to this child |
64
|
6
|
|
|
|
|
29
|
$self->set_container($self->tools->{entry_factory}->create_entry_for($self->{dir_name})); |
65
|
|
|
|
|
|
|
} |
66
|
6
|
|
|
|
|
62
|
return $self->{container}; |
67
|
|
|
|
|
|
|
} |
68
|
|
|
|
|
|
|
|
69
|
|
|
|
|
|
|
sub containers |
70
|
|
|
|
|
|
|
{ |
71
|
0
|
|
|
0
|
0
|
0
|
my $self = shift; |
72
|
0
|
|
|
|
|
0
|
return ($self->container->containers, $self->container); |
73
|
|
|
|
|
|
|
} |
74
|
|
|
|
|
|
|
|
75
|
|
|
|
|
|
|
sub neighbours |
76
|
|
|
|
|
|
|
{ |
77
|
0
|
|
|
0
|
0
|
0
|
my $self = shift; |
78
|
|
|
|
|
|
|
# TODO this should in theory use container method rather than the hash |
79
|
|
|
|
|
|
|
# element, but because this method needs element_index to be available |
80
|
|
|
|
|
|
|
# (which is only set in the entity if it is instantiated by its |
81
|
|
|
|
|
|
|
# container), then this will break in mysterious ways if the container |
82
|
|
|
|
|
|
|
# method has to instantiate the container object |
83
|
0
|
0
|
|
|
|
0
|
return (undef, undef) unless $self->{container}; |
84
|
0
|
|
|
|
|
0
|
return $self->{container}->neighbours_of_index($self->element_index); |
85
|
|
|
|
|
|
|
} |
86
|
|
|
|
|
|
|
|
87
|
|
|
|
|
|
|
# Returns the best available approximation of creation time of this entry |
88
|
|
|
|
|
|
|
sub creation_time |
89
|
|
|
|
|
|
|
{ |
90
|
10
|
|
|
10
|
0
|
5439
|
my $self = shift; |
91
|
10
|
|
|
|
|
19
|
my $stat = $self->{stat}; |
92
|
|
|
|
|
|
|
# We might not be able to get stat information (broken symlink, no permissions, ...) |
93
|
10
|
100
|
|
|
|
41
|
return undef unless $stat; |
94
|
|
|
|
|
|
|
# We need to use st_mtime, for lack of anything better |
95
|
4
|
|
|
|
|
30
|
return $stat->mtime; |
96
|
|
|
|
|
|
|
} |
97
|
|
|
|
|
|
|
|
98
|
|
|
|
|
|
|
sub content_modification_time |
99
|
|
|
|
|
|
|
{ |
100
|
0
|
|
|
0
|
0
|
0
|
my $self = shift; |
101
|
0
|
|
|
|
|
0
|
return $self->App::MaMGal::Entry::creation_time(@_); |
102
|
|
|
|
|
|
|
} |
103
|
|
|
|
|
|
|
|
104
|
|
|
|
|
|
|
sub fresher_than_me |
105
|
|
|
|
|
|
|
{ |
106
|
0
|
|
|
0
|
0
|
0
|
my $self = shift; |
107
|
0
|
|
|
|
|
0
|
my $path = shift; |
108
|
0
|
|
|
|
|
0
|
my %opts = @_; |
109
|
0
|
0
|
|
|
|
0
|
my $stat = stat($path) or return 0; |
110
|
0
|
0
|
|
|
|
0
|
return 1 if $stat->mtime >= $self->content_modification_time(%opts); |
111
|
0
|
|
|
|
|
0
|
return 0; |
112
|
|
|
|
|
|
|
} |
113
|
|
|
|
|
|
|
|
114
|
|
|
|
|
|
|
# Whether this entry should be shown in a directory contents montage |
115
|
3
|
|
|
3
|
0
|
2654
|
sub is_interesting { } |
116
|
|
|
|
|
|
|
|
117
|
|
|
|
|
|
|
# Some constants |
118
|
|
|
|
|
|
|
our $slides_dir = '.mamgal-slides'; |
119
|
6
|
|
|
6
|
0
|
3513
|
sub slides_dir { $slides_dir } |
120
|
6
|
|
|
6
|
0
|
3247
|
sub thumbnails_dir { '.mamgal-thumbnails' } |
121
|
0
|
|
|
0
|
0
|
0
|
sub medium_dir { '.mamgal-medium' } |
122
|
|
|
|
|
|
|
|
123
|
|
|
|
|
|
|
####################################################################################################################### |
124
|
|
|
|
|
|
|
# Abstract methods |
125
|
|
|
|
|
|
|
# these two need to return the text of the link ... |
126
|
2
|
|
|
2
|
0
|
2877
|
sub page_path { croak(sprintf("INTERNAL ERROR: Class [%s] does not define page_path.", ref(shift))) } |
127
|
2
|
|
|
2
|
0
|
2305
|
sub thumbnail_path { croak(sprintf("INTERNAL ERROR: Class [%s] does not define thumbnail_path.", ref(shift))) } |
128
|
|
|
|
|
|
|
|
129
|
|
|
|
|
|
|
1; |