summaryrefslogtreecommitdiffstats
path: root/scripts/pangotext.pl
diff options
context:
space:
mode:
Diffstat (limited to 'scripts/pangotext.pl')
-rw-r--r--scripts/pangotext.pl253
1 files changed, 253 insertions, 0 deletions
diff --git a/scripts/pangotext.pl b/scripts/pangotext.pl
new file mode 100644
index 0000000..125a538
--- /dev/null
+++ b/scripts/pangotext.pl
@@ -0,0 +1,253 @@
+##############################################################################################
+# pangotext.pl - Specify color patterns in a text message using html-esque tags. In the
+# same vein as pangotext.
+#
+# DESCRIPTION
+# The purpose of this script is to allow you to write text to the current channel with
+# complex color patterns, using simple html tag syntax. This allows you to do things like
+# send a rainbow-colored message where only part of the message is rainbow colored easily.
+#
+# USAGE
+# /pango <message text here>
+# NOTE: You can't put tags inside tags marked 'no inner tags' below
+# all other tags are fully nestable. Tags noted below with 'attributes'
+# also have attributes that can be specified (ie <tag attrib=value>) no spaces
+# are allowed in attribute names or values.
+# inverse,inv Reverse foreground and background of text
+# bold,b Bold text
+# underline,ul Underlines text
+# rainbow,rb Colorizes text with a rainbow (no inner tags)
+# checker Colorizes text with a checker pattern (no inner tags)
+# gradiant Colorizes text with a gradiant (no inner tags, attribs { start, end })
+# ...more to if you can think of any add more functions...
+#
+# EXAMPLES
+# This script makes most sense if you just use it and see how awesome it is. Here
+# are some example usages you should check out.
+# # Send a message with a colorful rainbow
+# /pango Hi guys, here's a <rainbow>rainbow</rainbow> for you.
+# /pango Hi guys, here's a <inverse><rainbow>rainbow</rainbow></inverse> for you. # Shows an inverse rainbow
+# /pango Hi guys, here's a <bold><rainbow>rainbow</rainbow></bold> for you. # Shows a bright rainbow
+#
+# # Send a message with a checker pattern and a rainbow and underlined text also
+# /pango <b>Let's play a game</b> <ul>of</ul> <checker>checkers</checker>! Or do you like
+# it better inversed <inverse><checker>inversed checkers</checker></inverse>!
+#
+# # Gradiants allow start and end range specifier in by color name:
+# /pango <gradiant start=green end=red>some gradiant text here</gradiant>
+# /pango <gradiant>default gradiant range</gradiant>
+# /pango <gradiant start=lightcyan end=white>a light gradiant</gradiant>
+#
+##############################################################################################
+use strict;
+use warnings;
+
+use Irssi;
+use Irssi::Irc;
+use utf8;
+
+our $VERSION = "1.2";
+our %IRSSI = (
+ authors => 'fprintf',
+ contact => 'fprintf@github.com',
+ name => 'pangotext',
+ description => 'Render text with various color modifications using HTML tag syntax.',
+ license => 'GNU GPLv2 or later',
+);
+
+# Color metadata
+my %color = (
+ white => 0,
+ black => 1,
+ blue => 2,
+ green => 3,
+ lightred => 4,
+ red => 5,
+ purple => 6,
+ orange => 7,
+ yellow => 8,
+ lightgreen => 9,
+ cyan => 10,
+ lightcyan => 11,
+ lightblue => 12,
+ lightpurple => 13,
+ gray => 14,
+ lightgray => 15,
+);
+
+my @color_order = (
+ 'white', 'lightgray', 'lightcyan', 'lightblue', 'lightgreen',
+ 'lightpurple', 'yellow', 'lightred', 'orange', 'red', 'purple',
+ 'cyan', 'blue', 'green', 'gray', 'black'
+);
+my %color_ordermap;
+for (my $i = 0; $i < @color_order; ++$i) {
+ $color_ordermap{$color_order[$i]} = $i;
+}
+
+# Allowed tags
+my %tag_registry = (
+ 'rb' => \&rainbow,
+ 'rainbow' => \&rainbow,
+ 'checker' => \&checker,
+
+ 'gradiant' => \&gradiant,
+ 'gradient' => \&gradiant,
+ 'grad' => \&gradiant,
+
+ 'ul' => \&underline,
+ 'underline' => \&underline,
+ 'bold' => \&bold,
+ 'b' => \&bold,
+ 'inverse' => \&inverse,
+ 'inv' => \&inverse,
+);
+
+my $utf8;
+
+##############################################################################################
+# Utils
+##############################################################################################
+
+sub palettize
+{
+ my ($text, $palette) = @_;
+ return $text if (!$palette || ref($palette) ne 'ARRAY');
+
+ # Colorize the text using the given palette
+ my $count = 0;
+ my $render = '';
+ foreach my $let (split(//,$text)) {
+ $let .= ',' if ($let eq ',');
+ $render .= $let =~ /\s/ ? $let : sprintf("\003%02d%s", $$palette[$count++ % scalar(@$palette)], $let);
+ }
+ return sprintf("%s\003", $render);
+}
+
+##############################################################################################
+# Render tags
+##############################################################################################
+sub rainbow
+{
+ my $text = shift;
+ my @palette = (
+ 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13
+ );
+ return palettize($text, \@palette);
+}
+
+sub gradiant
+{
+ my ($text, $attribs) = @_;
+ $attribs ||= {};
+ $attribs->{start} ||= 'white';
+ $attribs->{end} ||= 'lightpurple';
+
+ # Build the palette based on the given color range
+ my @palette = ();
+ my ($start,$end) = ($color_ordermap{$attribs->{start}},$color_ordermap{$attribs->{end}});
+ # Fancy way to find min and max
+ my $min = ($start,$end)[$start > $end];
+ my $max = ($start,$end)[$start < $end];
+ for (my $i = $min; $i <= $max; ++$i) {
+ push(@palette, $color{$color_order[$i % scalar(@color_order)]}); # Wrap colors around if they overlap
+ }
+
+ # Palettize the text
+ return palettize($text, \@palette);
+}
+
+sub checker
+{
+ my $text = shift;
+ my $rainbow = '';
+ my $count = 0;
+ # Black on red, red on black
+ my @palette = ('01,04', '04,01');
+ foreach my $let (split(//,$text)) {
+ $let .= ',' if ($let eq ',');
+ $rainbow .= $let =~ /\s/ ? $let : sprintf("\003%s%s", $palette[$count++ % scalar(@palette)], $let);
+ }
+ return sprintf("%s\003", $rainbow);
+}
+
+sub bold
+{
+ my $text = shift;
+ return sprintf("\002%s\002", $text);
+}
+
+sub underline
+{
+ my $text = shift;
+ return sprintf("\037%s\037", $text);
+}
+
+sub inverse
+{
+ my $text = shift;
+ return sprintf("\026%s\026", $text);
+}
+
+##############################################################################################
+# Renderer function
+##############################################################################################
+
+sub render
+{
+ my ($text) = @_;
+
+ while ($text =~ /<\s*([^>\s]+)\s*([^>]*)>(.+?)<\/?\1>/g) {
+ my ($action,$extra,$msg) = ($1,$2,$3);
+ my $mstart = $-[0];
+ my $mend = pos($text);
+ my %attribs = ();
+
+ (%attribs) = $extra =~ /(\S+)\s*=\s*(\S+)/g;
+
+ if (!exists($tag_registry{$action})) {
+ Irssi::print("[/pango error] invalid action: $action");
+ next;
+ }
+
+ # Render our text
+ $msg = $tag_registry{$action}->($msg,\%attribs);
+ my $len = $mend - $mstart;
+ my $index = $mend - $len;
+ # Insert it
+ substr($text, $index, $len, $msg);
+ }
+ return $text;
+}
+
+##############################################################################################
+# Irssi interface
+##############################################################################################
+# /pango
+# Send message to current channel
+# with rendered text
+# See functions above for available tags
+sub pango {
+ my ($text, $server, $dest) = @_;
+
+ if (!$server || !$server->{connected}) {
+ Irssi::print("[/pango error] not connected to server");
+ return;
+ }
+
+ return unless $dest;
+ if ($utf8) {
+ utf8::decode($text)
+ }
+
+ if ($dest->{type} eq "CHANNEL" || $dest->{type} eq "QUERY") {
+ $dest->command("/msg " . $dest->{name} . " " . render($text));
+ }
+}
+
+
+Irssi::command_bind("pango", \&pango);
+
+$utf8= Irssi::settings_get_str('term_charset') eq 'UTF-8';
+
+# vim:set ts=4 sw=4 expandtab: