chiark / gitweb /
Minimum trade value feature
[ypp-sc-tools.db-test.git] / yarrg / web / lookup
old mode 100644 (file)
new mode 100755 (executable)
index e062e60..b99cf0a
@@ -1,17 +1,43 @@
-<html><head><title>Route - YARRG</title></head><body>
+<%doc>
 
-<a href="<% $m->current_comp()->name() %>">YARRG</a> -
- Yet Another Revenue Research Gatherer
-<p>
+ This is part of the YARRG website.  YARRG is a tool and website
+ for assisting players of Yohoho Puzzle Pirates.
+
+ Copyright (C) 2009 Ian Jackson <ijackson@chiark.greenend.org.uk>
+ Copyright (C) 2009 Clare Boothby
+
+  YARRG's client code etc. is covered by the ordinary GNU GPL (v3 or later).
+  The YARRG website is covered by the GNU Affero GPL v3 or later, which
+   basically means that every installation of the website will let you
+   download the source.
+
+ This program is free software: you can redistribute it and/or modify
+ it under the terms of the GNU Affero General Public License as
+ published by the Free Software Foundation, either version 3 of the
+ License, or (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ GNU Affero General Public License for more details.
+
+ You should have received a copy of the GNU Affero General Public License
+ along with this program.  If not, see <http://www.gnu.org/licenses/>.
 
+ Yohoho and Puzzle Pirates are probably trademarks of Three Rings and
+ are used without permission.  This program is not endorsed or
+ sponsored by Three Rings.
+
+
+ This Mason component generates the main `lookup' page, including
+ all the entry boxes etc. for every query.
+
+
+</%doc>
 <%perl>
-my %a;
+my %ahtml;
 my @vars;
-
-# for output:
-my @archipelagoes;
-my @islandids;
-my %islandid2;
+my %styles;
 
 #---------- "mode" argument parsing and mode menu at top of page ----------
 
@@ -27,6 +53,42 @@ my %islandid2;
                CmpCanon => sub { !!$_[0] },
                Values => [     [ 0, 'Type in names' ],
                                [ 4, 'Select from menus' ] ]
+       }, {    Name => 'Query',
+               Before => 'Query: ',
+               Values => [     [ 'route', 'Trades for route' ],
+                               [ 'commod', 'Prices for commodity' ],
+                               [ 'offers', 'Offers at location' ],
+                               [ 'routesearch', 'Find profitable route' ],
+                               [ 'age', 'Data age' ] ]
+       }, {    Name => 'BuySell',
+               Before => '',
+               Values => [     [ 'buy_sell',   'Buy and sell' ],
+                               [ 'sell_buy',   'Sell and buy' ],
+                               [ 'buy',        'Buy offers only' ],
+                               [ 'sell',       'Sell offers only' ],
+                       ],
+               QuerySpecific => 1,
+       }, {    Name => 'ShowBlank',
+               Before => '',
+               Values => [     [ 0, 'Omit islands with no offers' ],
+                               [ 'show', 'Show all islands' ],
+                       ],
+               QuerySpecific => 1,
+       }, {    Name => 'ShowStalls',
+               Before => '',
+               Values => [     [ 0, 'Show total quantity at each price' ],
+                               [ 1, 'Show individual stalls' ],
+                               [ 2, "Also be cautious about stalls'".
+                                       " poe reserves",
+                                       '<a href="docs#poelimit">[?]</a>' ]
+                       ],
+               QuerySpecific => 1,
+       }, {    Name => 'RouteSearchType',
+               Before => 'Type of routes to search for: ',
+               Values => [     [ 0, 'Open-ended' ],
+                               [ 1, 'Circular' ],
+                       ],
+               QuerySpecific => 1,
        });
 
 foreach my $var (@vars) {
@@ -39,38 +101,101 @@ foreach my $var (@vars) {
                $val= [ $val, encode_entities($val) ];
        }
        if (exists $ARGS{$lname}) {
-               $a{$name}= $ARGS{$lname};
+               $styles{$name}= $ARGS{$lname};
+               my @html= grep { $_->[0] eq $styles{$name} }
+                               @{ $var->{Values} };
+               $ahtml{$name}= @html==1 ? $html[0][1] : '???';
        } else {
-               $a{$name}= $var->{Values}[0][0];
+               $styles{$name}= $var->{Values}[0][0];
+               $ahtml{$name}= $var->{Values}[0][1];
        }
 }
 
+</%perl>
+
+<%shared>
 my %baseqf;
+my %queryqf;
+</%shared>
+
+<%method formhidden>
+<%args>
+$ours
+</%args>
+% foreach my $n (keys %baseqf, keys %queryqf) {
+%      next if $ours->($n);
+%      my $v= exists $baseqf{$n} ? $baseqf{$n} : $queryqf{$n};
+<input type=hidden name=<% $n %> value="<% $v |h %>">
+% }
+</%method>
+
+<html lang="en"><head><title><% ucfirst $ahtml{Query} %> - YARRG</title>
+<style type="text/css">
+<& style.css &>
+  body { margin: 0; color: #000000; background: #c5c7ae; }
+  div.query { background: #d7d8b3; padding: 1em; }
+  div.results { padding: 1em; }
+  table.data { background: #b5b686; }
+  tr.datarow0 { background: #ffffff; }
+  tr.datarow1 { background: #e3e3e3; }
+</style>
+<&| script &>
+  function register_onload(f) {
+    var previous_onload= window.onload;
+    window.onload= function() {
+      if (previous_onload) previous_onload();
+      f();
+    };
+  }
+</&script>
+</head><body>
+
+% if (!printable($m)) {
+<div class="navoptbar">
+<& navbar &>
+<p>
+% }
+<%perl>
+
 foreach my $var (@vars) {
        my $lname= lc $var->{Name};
        next unless exists $ARGS{$lname};
        $baseqf{$lname}= $ARGS{$lname};
 }
 
-my %queryqf;
 foreach my $var (keys %ARGS) {
        next unless $var =~
-               m/^(?:routestring|islandid\d|archipelago\d|debug)$/;
-       $queryqf{$var}= $ARGS{$var};
+               m/^(?: (?:route|commod|capacity|capital|minprofit
+                        |island)string |
+                       lossperleague | distance |
+                       commodid |
+                       islandid \d |
+                       archipelago \d |
+                       debug |
+                       [RT]\w+
+                   )$/x;
+       my $val= $ARGS{$var};
+       next if $val eq 'none';
+       $queryqf{$var}= $val;
 }
 
-my $uri= URI->new($m->current_comp()->name());
-my $quri= sub { $uri->query_form(@_); $uri->path_query(); };
+my $quri= sub {
+       my $uri= URI->new('lookup');
+       $uri->query_form(@_);
+       $uri->path_query();
+};
 
-foreach my $var (@vars) {
+my $prselector_core= sub {
+       my ($var)= @_;
+       return if printable($m);
        my $name= $var->{Name};
        my $lname= lc $var->{Name};
        my $delim= $var->{Before};
-       my $canon= &{$var->{CmpCanon}}($a{$name});
+       my $canon= &{$var->{CmpCanon}}($styles{$name});
        my $cvalix= 0;
        foreach my $valr (@{ $var->{Values} }) {
                print $delim;  $delim= "\n|\n";
-               my ($value,$html) = @$valr;
+               my ($value,$html,$finally) = @$valr;
                my $iscurrent= &{$var->{CmpCanon}}($value) eq $canon;
                my $after;
                if ($iscurrent) {
@@ -79,271 +204,105 @@ foreach my $var (@vars) {
                } else {
                        my %qf= (%baseqf,%queryqf);
                        delete $qf{$lname};
+                       delete $qf{$_} foreach grep { m/^[A-Z]/ } keys %qf;
                        $qf{$lname}= $value if $cvalix;
-                       print '<a href="',$quri->(%qf),'">';
+</%perl>
+<a href="<% $quri->(%qf) |h %>">
+<%perl>
                        $after= '</a>';
                }
                print $html, $after;
+               print " ", $finally if defined $finally;
                $cvalix++;
        }
        print "<p>\n\n";
-}
-
-#---------- initial checks, startup, main entry form ----------
-
-dbw_connect($a{Ocean});
-
-</%perl>
-<%args>
-$debug => 0
-$routestring => ''
-</%args>
-
-<h1>Specify route</h1>
-<form action="<% $quri->() %>" method="get">
-
-%#---------- textbox, user enters route as string ----------
-% if (!$a{Dropdowns}) {
-
-Enter route (islands, or archipelagoes, separated by |s or commas;
- abbreviations are OK):<br/>
-
-<script type="text/javascript">
-tr_uri= "routetextstring?format=json&type=text/xml"
-               + "&ocean=<% uri_escape($a{Ocean}) %>";
-
-tr_timeout=false;
-tr_request=false;
-tr_done='';
-tr_needed='';
-function tr_Later(){
-  window.clearTimeout(tr_timeout);
-  tr_timeout = window.setTimeout(tr_Needed, 500);
-}
-function tr_Needed(){
-  window.clearTimeout(tr_timeout);
-  tr_element= document.getElementById('routestring');
-  tr_needed= tr_element.value;
-  tr_Request();
-}
-function tr_Request(){
-  if (tr_request || tr_needed==tr_done) return;
-  tr_done= tr_needed;
-  tr_request= new XMLHttpRequest();
-  uri= tr_uri+'&string='+encodeURIComponent(tr_needed);
-  tr_request.open('GET', uri);
-  tr_request.onreadystatechange= tr_Ready;
-  tr_request.send(null);
-}
-function tr_Ready() {
-  if (tr_request.readyState != 4) return;
-  if (tr_request.status == 200) {
-    response= tr_request.responseText;
-    eval('results='+response);
-    toedit= document.getElementById('routeresults');
-    toedit.innerHTML= results.show;
-  }
-  tr_request= false;
-  tr_Request();
-}
-window.onload= tr_Needed;
-</script>
-
-<input type="text" id="routestring" name="routestring" size=80
- value="<% $routestring |h %>"
- onchange="tr_Needed();"
- onkeyup="tr_Later();"><br>
-<div id="routeresults">&nbsp;</div><br/>
-
-% } else { #---------- dropdowns, user selects from menus ----------
+};
 
-<%perl>
-my ($sth,$row);;
-my @archlistdata;
-my %islandlistdata;
-$islandlistdata{'none'}= [ [ "none", "Select island..." ] ];
-
-my $optionlistmap= sub {
-       my ($optlist, $selected) = @_;
-       my $out='';
-       foreach my $entry (@$optlist) {
-               $out.= sprintf('<option value="%s" %s>%s</option>',
-                       encode_entities($entry->[0]),
-                       defined $selected && $entry->[0] eq $selected
-                               ? 'selected' : '',
-                       encode_entities($entry->[1]));
+my $prselector= sub {
+       my ($name)= @_;
+       foreach my $var (@vars) {
+               if ($var->{Name} eq $name) {
+                       $prselector_core->($var);
+                       return;
+               }
        }
-       return $out;
+       die $name;
 };
 
-my $dbh= dbw_connect($a{Ocean});
-
-$sth= $dbh->prepare("SELECT DISTINCT archipelago FROM islands
-                           ORDER BY archipelago;");
-$sth->execute();
-
-while ($row=$sth->fetchrow_arrayref) {
-       my ($arch)= @$row;
-       push @archlistdata, [ $arch, $arch ];
-       $islandlistdata{$arch}= [ [ "none", "Whole arch" ] ];
-}
-
-$sth= $dbh->prepare("SELECT islandid,islandname,archipelago
-                            FROM islands
-                           ORDER BY islandname;");
-$sth->execute();
-
-while ($row=$sth->fetchrow_arrayref) {
-       my $arch= $row->[2];
-       push @{ $islandlistdata{'none'} }, [ @$row ];
-       push @{ $islandlistdata{$arch} }, [ @$row ];
-       $islandid2{$row->[0]}= { Name => $row->[1], Arch => $arch };
-}
-
-my %resetislandlistdata;
-foreach my $arch (keys %islandlistdata) {
-       $resetislandlistdata{$arch}=
-               $optionlistmap->($islandlistdata{$arch}, '');
-}
-
-</%perl>
-
-<input type=hidden name=dropdowns value="<% $a{Dropdowns} %>">
-
-<script type="text/javascript">
-ms_lists= <% to_json(\%resetislandlistdata) %>;
-function ms_Setarch(dd) {
-  debug('ms_SetArch '+dd+' arch='+arch);
-  var arch= document.getElementsByName('archipelago'+dd).item(0).value;
-  var got= ms_lists[arch];
-  if (got == undefined) return; // unknown arch ?  hrm
-  debug('ms_SetArch '+dd+' arch='+arch+' got ok');
-  var select= document.getElementsByName('islandid'+dd).item(0);
-  select.innerHTML= got;
-  debug('ms_SetArch '+dd+' arch='+arch+' innerHTML set');
+foreach my $var (@vars) {
+       next if $var->{QuerySpecific};
+       $prselector_core->($var);
 }
-</script>
-
-<table style="table-layout:fixed; width:90%;">
-
-<tr>
-%      for my $dd (0..$a{Dropdowns}-1) {
-<td>
-<select name="archipelago<% $dd %>" onchange="ms_Setarch(<% $dd %>)">
-<option value="none">Whole ocean</option>
-<% $optionlistmap->(\@archlistdata, $ARGS{"archipelago$dd"}) %></select></td>
-%      }
-</tr>
-
-<tr>
-%      for my $dd (0..$a{Dropdowns}-1) {
-%              my $arch= $ARGS{"archipelago$dd"};
-%              $arch= 'none' if !defined $arch;
-<td>
-<select name="islandid<% $dd %>">
-<% $optionlistmap->($islandlistdata{$arch}, $ARGS{"islandid$dd"}) %>
-</select></td>
-%      }
-</tr>
-
-</table>
-
-% } #---------- end of dropdowns, now common middle of page code ----------
-
-<input type=submit name=submit value="Go">
-</form>
 
-<%perl>
-#========== result computations ==========
-
-my $results_head;
-$results_head= sub {
-       print "<h1>Results</h1>\n";
-       $results_head= sub { };
-};
+#---------- initial checks, startup, main entry form ----------
 
-#---------- result computation - textstring ----------
-if (!$a{Dropdowns}) {
-  if (length $routestring) {
-       $results_head->();
-       my $rsr= $m->comp('routetextstring',
-               ocean => $a{Ocean},
-               string => $routestring,
-               format => 'return'
-       );
-       if (length $rsr->{Error}) {
-               print encode_entities($rsr->{Error});
-       } else {
-               foreach my $entry (@{ $rsr->{Results} }) {
-                       push @archipelagoes,
-                               defined $entry->[1] ? undef : $entry->[0];
-                       push @islandids, $entry->[1];
-               } 
-       }
-  }
+die if $styles{Query} =~ m/[^a-z]/;
 
-} else { #---------- results - dropdowns ----------
+my $mydbh;
+my $dbh= ($mydbh= dbw_connect($styles{Ocean}));
 
-my $argorundef= sub {
-       my ($dd,$base) = @_;
-       my $thing= $ARGS{"${base}${dd}"};
-       $thing= undef if defined $thing and $thing eq 'none';
-       return $thing;
+my $results_head_done=0;
+my $someresults= sub {
+       return if $results_head_done;
+       $results_head_done=1;
+       my ($h)= @_;
+       $h= 'Results' if !$h;
+       print "\n<h1>$h</h1>\n";
 };
 
-for my $dd (0..$a{Dropdowns}-1) {
-       my $arch= $argorundef->($dd,'archipelago');
-       my $island= $argorundef->($dd,'islandid');
-       next unless defined $arch or defined $island;
-       if (defined $island and defined $arch) {
-               my $ii= $islandid2{$island};
-               my $iarch= $ii->{Arch};
-               if ($iarch ne $arch) {
-                       $results_head->();
 </%perl>
- Specified archipelago <% $arch %> but
- island <% $ii->{Name} %>
- which is in <% $iarch %>; using the island.<br>
-<%perl>
-               }
-               $arch= undef;
-       }
-       push @archipelagoes, $arch;
-       push @islandids, $island;
-}
-
-}#---------- result processing, common stuff
-</%perl>
-
-% if (@islandids) {
-%      $results_head->();
-
-<& routetrade, islandids => \@islandids, archipelagoes => \@archipelagoes &>
+<%args>
+$debug => 0
+</%args>
 
+% if (!printable($m)) {
+</div>
+<hr>
 % }
 
+<& "query_$styles{Query}", %baseqf, %queryqf, %styles,
+    quri => $quri, dbh => $dbh,
+    baseqf => \%baseqf, queryqf => \%queryqf, allargs => \%ARGS,
+    prselector => $prselector,
+    someresults => $someresults,
+    emsgokorprint => sub {
+       my ($emsg) = @_;
+       return 1 unless defined $emsg and length $emsg;
+       $someresults->();
+       print $emsg;
+       return 0;
+    }
+ &>
+
 %#---------- debugging and epilogue ----------
 
 % if ($debug) {
-<p>
+<div class="results">
 <pre id="debug_log">
 Debug log:
 </pre>
+</div>
 % }
 
-<script type="text/javascript">
+<&| script &>
 function debug (m) {
 % if ($debug) {
   var node= document.getElementById('debug_log');
   node.innerHTML += "\n" + m + "\n";
 % }
 }
-</script>
+</&script>
+
+<& footer &>
 
 <%init>
 use CommodsWeb;
 use HTML::Entities;
 use URI::Escape;
-use JSON;
 
 </%init>
+<%cleanup>
+
+$mydbh->rollback() if $mydbh;
+
+</%cleanup>