=head1 SYNOPSIS
- Usage: setbg [-u|sage] [-ve|rbose] [-d|ebug] [-s|leep <n>] [-bgdir <bgdir>]
-
+ Usage: setbg [-u|sage] [-h|elp] [-ve|rbose] [-d|ebug] [-s|leep <n>]
+ [-bgdirs <bgdir> -bgdirs ...]
+
Where:
- -u|sage: Displays this usage
- -ve|rbose: Be verbose
- -d|ebug: Output debug messages
+ -u|sage: Displays this usage
+ -h|elp: Display full help
+ -ve|rbose: Be verbose
+ -d|ebug: Output debug messages
- -s|leep: Number of minutes to sleep between setting the background
- (Default: 1 hour)
- -b|gdir: Directory to scan for images (Default: /web/Pictures)
+ -s|leep: Number of minutes to sleep between setting the background
+ (Default: 1 hour)
+ -l|ockscreen: Change lockscreen backround (Default: False)
+ -b|gdirs: Directories to scan for images
=head1 DESCRIPTION
-This script sets the background image randomly based on images $imgDir. Note
-if this script is run again it senses that it was previously run and sends the
-previous script a SIGUSR2 which the script intrprets as "Change the background
-now", then exits.
+This script sets the background image randomly based on images found in bgdirs.
+Note if this script is run again it senses that it was previously run and sends
+the previous script a SIGUSR2 which the script intrprets as "Change the
+background now", then exits.
+
+Data is written to the following files:
+
+ ~/.setbg: Contains the filename of the current background image
+ ~/.setbg.hist Contains a history of all images displayed for this run
+ ~/.setbg.stats Contains statistical information for the current run
+
+Also note that this script will process a SIGUSR1 to mean "re-evaluate the
+contents of the bgdirs incase it has changed and display a new image". This is
+useful for script to be able to alert setbg that something has changed. For
+example, a script named rmbg might look at ~/.setbg to get the name of the
+current background image file and remove it then signal setbg with SIGUSR1 to
+have it re-evaluate the state of bgdirs.
+
+Finally setbg will perform the this re-evaluation at midnight everyday. This is
+useful because we point setbg to look at -bgdirs from Dropbox where Camera
+Uploads is included and new pictures can arrive everyday.
=cut
use FindBin;
use Getopt::Long;
use Proc::ProcessTable;
+use File::Spec;
use CGI qw/:standard/;
use lib "$FindBin::Bin/../lib";
+use Pod::Usage;
+
+use DateUtils;
use Display;
+use Logger;
use Utils;
-my $VERSION = '$Revision: 1.10 $';
+my $VERSION = '$Revision: 1.12 $';
($VERSION) = ($VERSION =~ /\$Revision: (.*) /);
-my $processes = new Proc::ProcessTable;
+my $processes = Proc::ProcessTable->new;
+my %opts = (
+ sleep => 60,
+ lockscreen => 0,
+ usage => sub { pod2usage },
+ help => sub { pod2usage (-verbose => 2)},
+ verbose => sub { set_verbose },
+ debug => sub { set_debug },
+);
-foreach my $process (@{$processes->table}) {
- if ($process->cmndline =~ /setbg/ and
- $process->pid != $$) {
- kill 12, $process->pid;
+my %totals;
- exit 0;
- } # if
-} # foreach
+sub displayStats () {
+ my $statsFile = Logger->new(
+ name => ".$FindBin::Script.stats",
+ path => $ENV{HOME},
+ extension => '',
+ );
-$0 = "$FindBin::Script " . join ' ', @ARGV;
+ $statsFile->log('At ' . localtime());
+ $statsFile->log('Sleep: ' . $opts{sleep});
+ $statsFile->log('Image directories:');
-verbose "$FindBin::Script v$VERSION";
+ for (my $i = 0; $i < scalar @{$opts{bgdirs}}; $i++) {
+ $statsFile->log("\t$opts{bgdirs}[$i]: $opts{bgdircnt}[$i]")
+ } # for
-my $sleep = 60 * 60;
-my @imgDir;
+ Stats \%totals, $statsFile;
-GetOptions (
- 'usage' => sub { Usage },
- 'verbose' => sub { set_verbose },
- 'debug' => sub { set_debug },
- 'sleep=i' => \$sleep,
- 'bgdir=s' => \@imgDir,
-) || Usage;
+ return;
+} # displayStats
-foreach (@imgDir) {
- error "$_ is not a directory", 1 unless -d $_;
-} # foreach
+sub fillPictures () {
+ my @images;
-# Using gsettings
-my $setbg = "gsettings";
-my $setbgOpts = "set org.gnome.desktop.background picture-uri \"file://";
+ $totals{bgdirs} = 0;
+
+ for (@{$opts{bgdirs}}) {
+ my ($status, @pics) = Execute "find \"$_/\" -type f";
+
+ chomp @pics;
+
+ push @images, grep(/jpg$|png$|gif$/i, @pics);
-my @images;
+ @pics = grep(/jpg$|png$|gif$/i, @pics);
-foreach (@imgDir) {
- my @pics = `find "$_" -type f -name "*.jpg"`;
- chomp @pics;
- push @images, @pics;
-} # foreach
+ push @{$opts{bgdircnt}}, scalar @pics;
+
+ $totals{bgdirs}++;
+ } # for
+
+ $totals{images} = scalar @images;
+
+ displayStats;
+
+ return @images;
+} # fillPictures
+
+sub writeHistory($) {
+ my ($msg) = @_;
+
+ open my $hist, '>>', "$ENV{HOME}/.$FindBin::Script.hist"
+ or error "Unable to open $ENV{HOME}/.$FindBin::Script.hist for append - $!", 1;
+
+ $msg = localtime() . $msg;
+
+ display $msg, $hist;
+
+ close $hist;
+} # writeHistory
+
+sub updateSetBG($$) {
+ my ($bgimage, $lockimage) = @_;
+
+ open my $setbg, '>', "$ENV{HOME}/.$FindBin::Script"
+ or error "Unable to open $ENV{HOME}/.$FindBin::Script for writing - $!", 1;
+
+ display $bgimage, $setbg;
+
+ close $setbg;
+
+ my $msg = ":$bgimage";
+ $msg .= " lock:$lockimage" if $opts{lockscreen};
+
+ writeHistory $msg;
+
+ return;
+} # updateSetBG
-Usage "No images to display. Must specify -imgDir" unless @images;
-
sub SwitchWallPaper {
# We don't need to do anything here, just handle the interrupt and
# let the while loop continue.
debug 'SwitchWallPaper: Interrupt received';
+ displayStats;
+
+ return;
} # SwitchWallPaper
+## Main
+verbose "$FindBin::Script v$VERSION";
+
+my @argvCopy = @ARGV;
+
+GetOptions (
+ \%opts,
+ 'usage',
+ 'help',
+ 'verbose',
+ 'debug',
+ 'sleep=i',
+ 'lockscreen',
+ 'bgdirs=s@',
+) || Usage;
+
+local $0 = "$FindBin::Script " . join ' ', @argvCopy;
+
+for my $process (@{$processes->table}) {
+ if ($process->cmndline =~ /setbg/ and
+ $process->pid != $$) {
+ kill 12, $process->pid;
+
+ exit 0;
+ } # if
+} # for
+
+for (my $i = 0; $i < scalar @{$opts{bgdirs}}; $i++) {
+ error "$opts{bgdirs}[$i] is not a directory", 1 unless -d $opts{bgdirs}[$i];
+
+ $opts{bgdirs}[$i] = File::Spec->rel2abs ($opts{bgdirs}[$i]);
+} # for
+
+# Using gsettings
+my $setbg = "gsettings";
+my $setbgOpts = "set org.gnome.desktop.background picture-uri \"file://";
+my $setLockOpts = "set org.gnome.desktop.screensaver picture-uri \"file://";
+
+my @images = fillPictures;
+
+Usage "No images to display. Must specify -bgdirs" unless @images;
+
$SIG{USR2} = \&SwitchWallPaper;
+$SIG{USR1} = \&fillPictures;
my $debugger = $DB::OUT;
+my $today;
+
+truncate "$ENV{HOME}/.$FindBin::Script.hist", 0;
EnterDaemonMode unless defined $DB::OUT;
+my $pickNewImages = 1;
+my ($bgimage, $lockimage);
+
while () {
- my $image = escapeHTML ($images[int (rand $#images)]);
+ if ($pickNewImages) {
+ $bgimage = escapeHTML ($images[int (rand $#images)]);
+ $lockimage = escapeHTML ($images[int (rand $#images)]);
+ } # if
+
+ my $monitorIsOn;
+
+ my ($status, @output) = Execute("xset q | grep Monitor | awk '{print \$3}'");
+
+ if ($status or $output[0] eq 'Off') {
+ writeHistory ":Monitor off, not setting background to $bgimage - will keep trying";
+
+ $pickNewImages = 0;
+ } else {
+ $pickNewImages = 1;
+
+ my $cmd = "$setbg $setbgOpts$bgimage\" 2> /dev/null";
+
+ ($status, @output) = Execute $cmd;
+
+ if ($status) {
+ error "Trying to set background - command used \"$cmd\"\n\nOutput\n\n" .
+ join "\n", @output;
+ $totals{errors}++;
+ } else {
+ $totals{'Images displayed'}++;
+ } # if
+
+ if ($opts{lockscreen}) {
+ $cmd = "$setbg $setLockOpts$lockimage\" 2> /dev/null";
+
+ ($status, @output) = Execute $cmd;
+
+ if ($status != 0) {
+ error "Trying to set lock screen - command used \"$cmd\"\n\nOutput\n\n" .
+ join "\n", @output;
+ $totals{errors}++;
+ } else {
+ $totals{'Lock screens displayed'}++;
+ } # if
+ } # if
+
+ updateSetBG $bgimage, $lockimage;
+ } # if
- open my $log, '>', "$ENV{HOME}/.$FindBin::Script"
- or error "Unable to open $ENV{HOME}/.setbg for writing - $!", 1;
+ displayStats;
- display $image, $log;
+ $today = YMD;
- my $cmd = "$setbg $setbgOpts$image\" 2> /dev/null";
+ sleep $opts{sleep} * 60;
- my @output = `$cmd`;
+ if ($today ne YMD){
+ @images = fillPictures;
- if ($? != 0) {
- display "ERROR Trying to set background - command used \"$cmd\"", $log;
- display "Output:";
- display join "\n", @output, $log;
+ displayStats;
} # if
-
- close $log;
-
- sleep $sleep;
} # while