Group
Extension

Mojolicious-Plugin-Notifications/lib/Mojolicious/Plugin/Notifications/JSON.pm

package Mojolicious::Plugin::Notifications::JSON;
use Mojo::Base 'Mojolicious::Plugin::Notifications::Engine';

# TODO:
#   Probably introduce a notify_json function

has key => 'notifications';

# Nothing to register
sub register {
  my ($plugin, $app, $param) = @_;
  $plugin->key($param->{key}) if $param->{key};
};


# Notification method
sub notifications {
  my ($self, $c, $notify_array, $rule, $json, %param) = @_;

  my $key = $param{key} // $self->key;

  return $json unless @$notify_array;

  if (!$json || ref $json) {
    my @msgs;
    foreach (@$notify_array) {

      # Confirmation message
      if (ref $_->[1] && ref $_->[1] eq 'HASH' &&
            ($_->[1]->{ok} || $_->[1]->{cancel} || $_[1]->{confirm})) {
        my $param = $_->[1];

        my $opt = {};

        # Set confirmation path
        if ($param->{confirm}) {
          $opt->{$param->{confirm_label} // 'confirm'} = {
            method => 'GET',
            url => $param->{confirm}
          };
        }

        else {
          # Set confirmation path
          if ($param->{ok}) {
            $opt->{$param->{ok_label} // 'ok'} = {
              method => 'POST',
              url => $param->{ok}
            };
          }

          # Set cancelation path
          if ($param->{cancel}) {
            $opt->{$param->{cancel_label} // 'cancel'} = {
              method => 'POST',
              url => $param->{cancel}
            };
          };
        };

        push @msgs, [$_->[0], ''.$_->[-1], $opt];
      }

      # Normal notification
      else {
        push(@msgs, [$_->[0], ''.$_->[-1]]);
      };
    };

    if (!$json) {
      return { $key => \@msgs }
    }

    # Obect is an array
    elsif (index(ref $json, 'ARRAY') >= 0) {
      push(@$json, { $key => \@msgs });
    }

    # Object is a hash
    elsif (index(ref $json, 'HASH') >= 0) {
      my $n = ($json->{$key} //= []);
      push(@$n, @msgs);
    };
  };

  return $json;
};


1;


__END__

=pod

=encoding utf8

=head1 NAME

Mojolicious::Plugin::Notifications::JSON - Event Notifications in JSON


=head1 SYNOPSIS

  # Register the engine
  plugin Notifications => {
    JSON => 1
  };

  # In the controller
  $c->render(json => $c->notifications(json => $json));


=head1 DESCRIPTION

This plugin is a simple notification engine for JSON.

If it does not suit your needs, you can easily
L<write your own engine|Mojolicious::Plugin::Notifications::Engine>.


=head1 METHODS

L<Mojolicious::Plugin::Notifications::JSON> inherits all methods
from L<Mojolicious::Plugin::Notifications::Engine> and implements
the following new one.


=head2 register

  plugin Notifications => {
    JSON => {
      key => 'notes'
    }
  };

Called when registering the main plugin.
All parameters under the key C<JSON> are passed to the registration.

Accepts the following parameters:

=over 4

=item B<key>

Define the attribute name of the notification array.
Defaults to C<notifications>.

=back


=head1 HELPERS

=head2 notify

  $c->notify(warn => 'wrong');
  $c->notify(confirm => { ok => '/ok', cancel => '/cancel' } => 'Please, confirm!');

  $c->render(json => $c->notifications(json => { text => 'html' }));
  # {
  #   "notifications":[
  #     ["warn", "wrong"],
  #     [
  #       "confirm",
  #       "Please, confirm!",
  #       {
  #         "cancel":{
  #           "method":"POST",
  #           "url":"\/cancel"
  #         },
  #         "ok":{
  #           "method":"POST",
  #           "url":"\/ok"
  #         }
  #       }
  #     ]
  #   ],
  #   "text":"example"
  # }


See the base L<notify|Mojolicious::Plugin::Notifications/notify> helper.

In case an C<ok>, C<cancel> or C<confirm> parameter is passed,
this will add an object containing confirmation and/or cancellation paths.
The C<confirm> URL should point to an (HTML) endpoint a user
can enter confirmation details. If it is given, both the C<ok> and C<cancel>
parameters will be ignored.
In case an C<ok_label> is passed, this will be the key for the
confirmation object.
In case a C<cancel_label> is passed, this will be the key
for the cancelation object.
In case a C<confirm_label> is passed, this will be the key
for the confirmation object.

Confirmation routes for JSON do not support CSRF protection.


B<Confirmation is EXPERIMENTAL!>

=head2 notifications

  $c->render(json => $c->notifications(json => $json));
  $c->render(json => $c->notifications(json => $json, key => 'notes'));

Merge notifications into your JSON response.

In case JSON is an object, it will inject an attribute
that points to an array reference containing the notifications.
If the JSON is an array, an object is appended with one attribute
that points to an array reference containing the notifications.
If the JSON is empty, an object will be created with one attribute
that points to an array reference containing the notifications.

If the JSON is not of one of the descripted types, it's returned
unaltered.

The name of the attribute can either be given on registration or
by passing a parameter for C<key>.
The name defaults to C<notifications>.


=head1 AVAILABILITY

  https://github.com/Akron/Mojolicious-Plugin-Notifications


=head1 COPYRIGHT AND LICENSE

Copyright (C) 2014-2018, L<Nils Diewald|http://nils-diewald.de/>.

This program is free software, you can redistribute it
and/or modify it under the terms of the Artistic License version 2.0.

=cut


Powered by Groonga
Maintained by Kenichi Ishigaki <ishigaki@cpan.org>. If you find anything, submit it on GitHub.