3 |
|
|
4 |
extends 'Frey'; |
extends 'Frey'; |
5 |
with 'Frey::Web'; |
with 'Frey::Web'; |
6 |
|
with 'Frey::Path'; |
7 |
|
|
8 |
|
use Moose::Util::TypeConstraints; |
9 |
|
|
10 |
|
enum 'SVK_Action' => ( 'commit', 'revert', 'postpone' ); |
11 |
|
|
12 |
|
has action => ( |
13 |
|
is => 'rw', |
14 |
|
isa => 'SVK_Action', |
15 |
|
); |
16 |
|
|
17 |
|
has path => ( |
18 |
|
documentation => 'path to work with', |
19 |
|
is => 'rw', |
20 |
|
isa => 'Str|ArrayRef', |
21 |
|
); |
22 |
|
|
23 |
|
has commit_message => ( |
24 |
|
documentation => 'commit message', |
25 |
|
is => 'rw', |
26 |
|
isa => 'Str', |
27 |
|
); |
28 |
|
|
29 |
sub svk { |
sub svk { |
30 |
my ( $self, $exec, $coderef ) = @_; |
my ( $self, $exec, $coderef ) = @_; |
36 |
close($svk) or die "can't close svk $exec: $@"; |
close($svk) or die "can't close svk $exec: $@"; |
37 |
} |
} |
38 |
|
|
39 |
|
our $svk_status_path = '^(\w+[\+\s]+)(.+)'; |
40 |
|
|
41 |
sub modified { |
sub modified { |
42 |
my ($self) = @_; |
my ($self) = @_; |
43 |
my @modified; |
my @modified; |
44 |
my $svk = $self->svk('status -q', sub { |
my $svk = $self->svk('status -q', sub { |
45 |
push @modified, $1 if /^(M|A)\s+(.+)/; |
push @modified, $2 if m{$svk_status_path}; |
46 |
}); |
}); |
47 |
return @modified; |
return @modified; |
48 |
} |
} |
55 |
my ( $label, $value ) = split(/:\s+/, $_, 2); |
my ( $label, $value ) = split(/:\s+/, $_, 2); |
56 |
$info->{$label} = $value if $label; |
$info->{$label} = $value if $label; |
57 |
}); |
}); |
58 |
|
warn "# svk info ",$self->dump( $info ); |
59 |
return $info; |
return $info; |
60 |
} |
} |
61 |
|
|
66 |
} |
} |
67 |
} |
} |
68 |
|
|
69 |
sub as_markup { |
sub commit_as_markup { |
70 |
my ($self) = @_; |
my ($self) = @_; |
|
|
|
71 |
my $status = `svk status -q`; |
my $status = `svk status -q`; |
72 |
|
$status =~ s{$svk_status_path}{$1 . $self->checkbox('path',$2) . qq|<a href="#$2">$2</a>|}egm; |
73 |
|
if ( $status ) { |
74 |
|
$self->add_css(qq| |
75 |
|
pre.l a { text-decoration: none; } |
76 |
|
form.commit { |
77 |
|
background: #eee; |
78 |
|
padding: 1em 1em; |
79 |
|
position: fixed; |
80 |
|
top: 1em; |
81 |
|
right: 1em; |
82 |
|
z-index: 10; |
83 |
|
opacity: .2; |
84 |
|
filter: alpha(opacity=20); |
85 |
|
} |
86 |
|
form.commit:hover { |
87 |
|
opacity: 1; |
88 |
|
filter: alpha(opacity=100); |
89 |
|
} |
90 |
|
| ); |
91 |
|
|
92 |
|
|
93 |
|
$status = qq| |
94 |
|
<form class="commit" method="post"> |
95 |
|
<pre class="l">$status</pre> |
96 |
|
<textarea name="commit_message" cols=40 rows=4></textarea> |
97 |
|
<br><input type="submit" name="action" value="commit"> |
98 |
|
</form> |
99 |
|
|; |
100 |
|
$self->add_status( status => $status ); |
101 |
|
warn "commit_as_markup ",length($status)," bytes"; |
102 |
|
} |
103 |
|
return $status; |
104 |
|
} |
105 |
|
|
106 |
|
sub diff_as_markup { |
107 |
|
my ($self) = @_; |
108 |
|
|
109 |
my $diff = `svk diff`; |
my $diff = `svk diff`; |
110 |
|
$self->add_status( diff => $diff ); |
111 |
|
|
112 |
|
$diff = $self->html_escape( $diff ); |
113 |
|
$self->add_css( qq| |
114 |
|
pre span.add { background: #dfd } |
115 |
|
pre span.del { background: #fdd } |
116 |
|
pre form.inline { display: inline } |
117 |
|
| ); |
118 |
|
$diff =~ s{^(\+.+?)$}{<span class="add">$1</span>}gm; |
119 |
|
$diff =~ s{^(\-.+?)$}{<span class="del">$1</span>}gm; |
120 |
|
sub form { |
121 |
|
my ( $path, $action ) = @_; |
122 |
|
qq|<form class="inline" method="post"><input type="hidden" name="path" value="$path"><input type="submit" name="action" value="$action"></form>|; |
123 |
|
}; |
124 |
|
$diff =~ s{^(===\s+)(\S+)$}{$1 . form($2,'revert') . qq| <a name="$2" target="editor" href="/editor+$2+1">$2</a> | . form($2,'postpone') }gem; |
125 |
|
|
126 |
|
$diff = qq|<pre>$diff</pre>| if $diff; |
127 |
|
warn "diff_as_markup ",length($diff)," bytes"; |
128 |
|
return $diff; |
129 |
|
} |
130 |
|
|
131 |
|
sub action_as_markup { |
132 |
|
my ($self) = @_; |
133 |
|
|
134 |
|
my $cmd; |
135 |
|
|
136 |
|
if ( $self->action eq 'postpone' ) { |
137 |
|
my $old = $self->path; |
138 |
|
my $new = $old; |
139 |
|
$new =~ s{/([^/]+)$}{/.postponed.$1}; |
140 |
|
|
141 |
|
die "Allready have ", $self->path_size($new) if -e $new; |
142 |
|
$cmd = "mv $old $new && svk revert $old"; |
143 |
|
} elsif ( $self->action ) { |
144 |
|
$cmd = 'svk ' . $self->action; |
145 |
|
if ( $self->action eq 'commit' ) { |
146 |
|
my $msg = $self->commit_message || return $self->error( "need commit message\n" ); |
147 |
|
$msg =~ s{"}{\\"}gs; |
148 |
|
$cmd .= qq{ -m "$msg"}; |
149 |
|
} else { |
150 |
|
confess "need path" unless $self->path; |
151 |
|
} |
152 |
|
|
153 |
|
my @paths = eval { @{ $self->path } }; # XXX sigh! |
154 |
|
@paths = ( $self->path ) unless @paths; |
155 |
|
warn "# path ", $self->dump( @paths ); |
156 |
|
|
157 |
|
$cmd .= ' ' . join( ' ',@paths ); |
158 |
|
} |
159 |
|
if ( $cmd ) { |
160 |
|
$cmd .= ' 2>&1'; |
161 |
|
warn "# cmd $cmd"; |
162 |
|
|
163 |
|
my $out = `$cmd`; |
164 |
|
warn "# output of $cmd is: $out"; |
165 |
|
|
166 |
|
return qq| |
167 |
|
Command <tt>$cmd</tt> produced output: |
168 |
|
<pre style="background: #ff8;">$out</pre> |
169 |
|
|; |
170 |
|
} |
171 |
|
|
172 |
|
} |
173 |
|
|
174 |
|
sub as_markup { |
175 |
|
my ($self) = @_; |
176 |
|
|
177 |
|
my $html = $self->action_as_markup; |
178 |
|
|
179 |
|
$self->title( 'svk' . ( $self->action ? ' - ' . $self->action : '' ) ); # XXX without this we get wrong icon and title |
180 |
|
|
181 |
Frey::Web->meta->apply( $self ) unless $self->can('html_escape'); |
$html .= $self->commit_as_markup . $self->diff_as_markup || |
182 |
|
qq|No changes in tracked files|; |
183 |
|
|
184 |
my $html |
warn "as_markup ",length($html)," bytes"; |
|
= qq|<pre>$status</pre><hr><pre>| |
|
|
. $self->html_escape( $diff ) |
|
|
. qq|</pre>| |
|
|
; |
|
|
warn "diff ",length($html)," bytes"; |
|
185 |
|
|
186 |
return $html; |
return $html; |
187 |
} |
} |