Changeset 18079


Ignore:
Timestamp:
Sep 20, 2012, 6:04:23 PM (12 years ago)
Author:
mistic100
Message:

feature 2751: Redesign web API, webservices explorer

Location:
trunk/tools
Files:
1 deleted
1 edited

Legend:

Unmodified
Added
Removed
  • trunk/tools/ws.htm

    r3282 r18079  
    1 <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01//EN" "http://www.w3.org/TR/html4/strict.dtd">
    2 <html>
     1<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01//EN"
     2   "http://www.w3.org/TR/html4/strict.dtd">
     3<html xmlns="http://www.w3.org/1999/xhtml" lang="en" dir="ltr">
    34<head>
    4 <title>PWG web service explorer</title>
    5 <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
    6 <script type="text/javascript" src="prototype.js" ></script>
    7 
    8 <script type="text/javascript">
    9 
    10 function setVisibility(id, vis)
     5  <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
     6  <title>Piwigo web API (web-services) explorer</title>
     7 
     8  <script type="text/javascript" src="https://ajax.googleapis.com/ajax/libs/jquery/1.8.1/jquery.min.js"></script>
     9 
     10  <style type="text/css">
     11  /* BEGIN CSS RESET
     12    http://meyerweb.com/eric/tools/css/reset
     13    v2.0 | 20110126 | License: none (public domain) */
     14  html, body, div, span, applet, object, iframe, h1, h2, h3, h4, h5, h6, p, blockquote, pre, a, abbr, acronym, address, big, cite, code,
     15  del, dfn, em, img, ins, kbd, q, s, samp, small, strike, strong, sub, sup, tt, var, b, u, i, center, dl, dt, dd, ol, ul, li,
     16  fieldset, form, label, legend, table, caption, tbody, tfoot, thead, tr, th, td, article, aside, canvas, details, embed,
     17  figure, figcaption, footer, header, hgroup, menu, nav, output, ruby, section, summary, time, mark, audio, video
     18  {margin:0;padding:0;border:0;font-size:100%;vertical-align:baseline;}
     19
     20  article, aside, details, figcaption, figure, footer, header, hgroup, menu, nav, section {display:block;}
     21  body {line-height:1.1;}
     22  blockquote, q {quotes:none;}
     23  blockquote:before, blockquote:after, q:before, q:after {content:'';content:none;}
     24  table {border-collapse:collapse;border-spacing:0;}
     25  /* END CSS RESET */
     26
     27  html {font-family:"Corbel","Lucida Grande","Verdana",sans-serif;color:#222;font-size:13px;}
     28
     29  a {color:#247EBF;text-decoration:none;}
     30  a:hover {color:#EB9C39;border-bottom-width:1px;border-style:dotted;
     31    text-shadow:1px 1px 0 #ddd;-moz-text-shadow:1px 1px 0 #ddd;-webkit-text-shadow:1px 1px 0 #ddd;
     32  }
     33  blockquote {border:1px solid #cdcdcd;background:#F9F9F9;padding:8px;}
     34  hr {margin:10px 30px;color:#fff;}
     35  ul {margin-left:25px;}
     36  p {margin:8px 0;}
     37
     38  h1 {color:#fff;font-size:26px;padding:10px 15px;
     39    text-shadow:1px 1px 0 #999;-moz-text-shadow:1px 1px 0 #999;-webkit-text-shadow:1px 1px 0 #999;
     40    background:#45484d;background:-moz-linear-gradient(top, #45484d 0%, #333333 100%);background:-webkit-gradient(linear, left top, left bottom, color-stop(0%,#45484d), color-stop(100%,#333333));background:-webkit-linear-gradient(top, #45484d 0%,#333333 100%);background:-o-linear-gradient(top, #45484d 0%,#333333 100%);background:-ms-linear-gradient(top, #45484d 0%,#333333 100%);background:linear-gradient(to bottom, #45484d 0%,#333333 100%);filter:progid:DXImageTransform.Microsoft.gradient( startColorstr='#45484d', endColorstr='#333333',GradientType=0 );
     41  }
     42  h2 {color:#fff;font-size:20px;padding:5px 10px;
     43    text-shadow:1px 1px 0 #555;-moz-text-shadow:1px 1px 0 #555;-webkit-text-shadow:1px 1px 0 #555;
     44    background:#f2a841;background:-moz-linear-gradient(top, #f2a841 0%, #ef6b13 100%);background:-webkit-gradient(linear, left top, left bottom, color-stop(0%,#f2a841), color-stop(100%,#ef6b13));background:-webkit-linear-gradient(top, #f2a841 0%,#ef6b13 100%);background:-o-linear-gradient(top, #f2a841 0%,#ef6b13 100%);background:-ms-linear-gradient(top, #f2a841 0%,#ef6b13 100%);background:linear-gradient(to bottom, #f2a841 0%,#ef6b13 100%);filter:progid:DXImageTransform.Microsoft.gradient( startColorstr='#f2a841', endColorstr='#ef6b13',GradientType=0 );
     45  }
     46  h2#errorWrapper {color:#F42C00;font-weight:normal;
     47    background:#eaeaea;background:-moz-linear-gradient(top, #eaeaea 0%, #afafaf 100%);background:-webkit-gradient(linear, lefttop, leftbottom, color-stop(0%,#eaeaea), color-stop(100%,#afafaf));background:-webkit-linear-gradient(top, #eaeaea 0%, #afafaf 100%);background:-o-linear-gradient(top, #eaeaea 0%, #afafaf 100%);background:-ms-linear-gradient(top, #eaeaea 0%, #afafaf 100%);background:linear-gradient(tobottom, #eaeaea 0%, #afafaf 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#eaeaea', endColorstr='#afafaf', GradientType=0);
     48  }
     49  h3 {display:inline-block;padding:5px 10px;color:#555;font-weight:bold;text-align:center;
     50    border-radius:8px 8px 0 0;-moz-border-radius:8px 8px 0 0;-webkit-border-radius:8px 8px 0 0;
     51    text-shadow:1px 1px 0 #bbb;-moz-text-shadow:1px 1px 0 #bbb;-webkit-text-shadow:1px 1px 0 #bbb;
     52    background:#f2f2f2;background:-moz-linear-gradient(top, #f2f2f2 0%, #cecece 100%);background:-webkit-gradient(linear, left top, left bottom, color-stop(0%,#f2f2f2), color-stop(100%,#cecece));background:-webkit-linear-gradient(top, #f2f2f2 0%,#cecece 100%);background:-o-linear-gradient(top, #f2f2f2 0%,#cecece 100%);background:-ms-linear-gradient(top, #f2f2f2 0%,#cecece 100%);background:linear-gradient(to bottom, #f2f2f2 0%,#cecece 100%);filter:progid:DXImageTransform.Microsoft.gradient( startColorstr='#f2f2f2', endColorstr='#cecece',GradientType=0 );
     53  }
     54
     55  #the_header {border-bottom:1px solid #cdcdcd;margin-bottom:1px;}
     56  #the_footer {background:#EAEAEA;border-top:1px solid #cdcdcd;padding:10px;clear:both;}
     57
     58  #the_methods {width:250px;float:left;border-style:solid;border-color:#cdcdcd;border-width:1px 1px 0 0;
     59    background-image:url();
     60  }
     61  #the_methods ul {font-size:1.1em;margin:5px 0 10px 10px;list-style:none;}
     62  #the_methods li:before {content:"\203A\00A0";font-weight:bold;color:#EB9C39;font-size:1.1em;}
     63  #the_methods li:hover:before {content:"\00A0\203A";}
     64
     65  #the_page {margin-left:252px;border-style:solid;border-color:#cdcdcd;border-width:1px 0 0 1px;}
     66  #the_content {padding:10px;}
     67
     68  #methodParams {display:inline-block;}
     69  #methodParams thead td {background:#DEE3E9;font-weight:bold;padding:2px 5px;}
     70  #methodParams td {padding:2px;border:1px solid #cdcdcd;vertical-align:middle;}
     71  #methodParams tbody tr:nth-child(even) {background:#f7f7f7;}
     72  #methodParams tbody tr td:first-child {font-family:monospace;font-size:0.95em;}
     73  #methodParams td.mini {width:0px;text-align:center;}
     74  #methodParams tfoot {font-size:0.95em;}
     75  #methodParams td.input {text-align:center;}
     76  #methodParams td.input input[type="text"] {width:97%;font-size:0.9em;background:#f7f7f7;border:1px solid #ccc;
     77    border-radius:2px;-moz-border-radius:2px;-webkit-border-radius:2px;
     78  }
     79  #methodParams td.input input[type="text"]:hover, #methodParams td.input input[type="text"]:focus {border-color:#C7E2F1;border-top-color:#96BCD7;background:#fff;}
     80
     81  #testForm {display:inline-block;margin-left:15px;}
     82  #testForm td {padding:2px 0;}
     83  #testForm tr:last-child td {padding:8px 0 5px 0;}
     84  #testForm blockquote {width:200px;}
     85 
     86  #introMessage {font-size:1.1em;}
     87  #urlForm {margin-bottom:10px;}
     88
     89  a.button {color:#fff;padding:3px 8px;border:1px solid #91bb5c;font-size:0.9em;margin-right:3px;display:inline-block;
     90    border-radius:5px;-moz-border-radius:5px;-webkit-border-radius:5px;
     91    text-shadow:1px 1px 0 #666;-moz-text-shadow:1px 1px 0 #666;-webkit-text-shadow:1px 1px 0 #666;
     92    background:#84bb3c;background:-moz-linear-gradient(top, #84bb3c 0%, #3f5a1d 100%);background:-webkit-gradient(linear, lefttop, leftbottom, color-stop(0%,#84bb3c), color-stop(100%,#3f5a1d));background:-webkit-linear-gradient(top, #84bb3c 0%, #3f5a1d 100%);background:-o-linear-gradient(top, #84bb3c 0%, #3f5a1d 100%);background:-ms-linear-gradient(top, #84bb3c 0%, #3f5a1d 100%);background:linear-gradient(tobottom, #84bb3c 0%, #3f5a1d 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#84bb3c', endColorstr='#3f5a1d', GradientType=0);
     93  }
     94  a.button:hover {color:#E5FF00;}
     95 
     96  #iframeWrapper {width:100%;height:300px;padding:3px 3px 20px 3px;background:#F9F9F9;border:1px solid #cdcdcd;overflow:hidden;position:relative;}
     97  iframe {width:100%;height:100%;background:#fff;}
     98  </style>
     99 
     100</head>
     101
     102<body>
     103
     104<div id="the_header">
     105  <h1>Piwigo web API (web-services) explorer</h1>
     106</div> <!-- the_header -->
     107
     108<div id="the_methods">
     109  <h2>Available methods</h2>
     110 
     111  <ul id="methodsList">
     112  </ul>
     113</div> <!-- the_methods -->
     114
     115<div id="the_page">
     116  <h2 id="methodName" style="display:none;"></h2>
     117  <h2 id="errorWrapper" style="display:none;"></h2>
     118 
     119  <div id="the_content">
     120    <form id="urlForm" style="display:none;">
     121      <input type="text" name="ws_url" size="60">
     122      <input type="submit" value="Go!">
     123    </form>
     124   
     125    <blockquote id="introMessage">
     126      <p>
     127        <b>API = Application Programming Interface.</b><br>
     128        This is the way other applications can communicate with Piwigo. This feature is also know as Web Services.
     129      </p>
     130
     131      <p>Examples:</p>
     132      <ul>
     133        <li>Wordpress (web blog software) can display random photos from a Piwigo gallery in its sidebar</li>
     134        <li>Lightroom (photo management software for desktop) can create albums and upload photos to Piwigo</li>
     135      </ul>
     136
     137      <p>
     138        This page lists all API methods available on your Piwigo installation, part of the Piwigo core or added by third-party plugins.
     139        For each method you can consult required and optional parameters, and even test them in direct live!
     140      </p>
     141
     142      <p>
     143        For more information you can consult our Wiki <a href="http://piwigo.org/doc/doku.php?id=dev:webapi:start" target="_blank">Piwigo Web API</a> and <a href="http://piwigo.org/forum" target="_blank">our forums</a>.
     144      </p>
     145    </blockquote> <!-- introMessage -->
     146
     147
     148    <form id="methodWrapper" style="display:none;">
     149      <div id="methodDescription" style="display:none;">
     150        <h3>Description</h3>
     151        <blockquote>
     152        </blockquote>
     153       
     154        <hr>
     155      </div> <!-- methodDescription -->
     156     
     157      <div id="methodParams">
     158        <h3>Method parameters</h3>
     159        <table>
     160          <thead>
     161            <tr>
     162              <td style="width:150px;">Name</td>
     163              <td class="mini">Extra</td>
     164              <td style="width:300px;">Value</td>
     165              <td class="mini">Send</td>
     166            </tr>
     167          </thead>
     168         
     169          <tbody>
     170          </tbody>
     171         
     172          <tfoot>
     173            <tr>
     174              <td colspan="4"><b>*</b>: required parameter, <b>?</b>: optional parameter, <b>[]</b>: parameter can be an array (use a pipe | to split values)</td>
     175            </tr>
     176          </tfoot>
     177        </table>
     178      </div> <!-- methodParams -->
     179     
     180      <div id="testForm">
     181        <h3>Test</h3>
     182        <blockquote>
     183          <table>
     184            <tr>
     185              <td>Request format :</td>
     186              <td>
     187                <select id="requestFormat">
     188                  <option value="get" selected="selected">GET</option>
     189                  <option value="post">POST</option>
     190                </select>
     191              </td>
     192            </tr>
     193            <tr>
     194              <td>Response format :</td>
     195              <td>
     196                <select id="responseFormat">
     197                  <option value="rest" selected="selected">REST (xml)</option>
     198                  <option value="json">JSON</option>
     199                  <option value="php">PHP serial</option>
     200                  <option value="xmlrpc">XML RPC</option>
     201                </select>
     202              </td>
     203            </tr>
     204            <tr>
     205              <td colspan="2">
     206                <a href="#" class="button" id="invokeMethod">INVOKE</a>
     207                <a href="#" class="button" id="invokeMethodBlank">INVOKE (new window)</a>
     208              </td>
     209            </tr>
     210          </table>
     211        </blockquote>
     212      </div> <!-- testForm -->
     213     
     214      <hr>
     215     
     216      <h3>Result</h3>
     217      <div id="iframeWrapper">
     218        <iframe src="" id="invokeFrame" name="invokeFrame"></iframe>
     219        <a href="#bottom" id="increaseIframe"><b>&darr;</b> increase height</a> &#8226; <a href="#bottom" id="decreaseIframe"><b>&uarr;</b> decrease height</a>
     220        <a name="bottom"></a>
     221      </div>
     222    </form> <!-- iframeWrapper -->
     223   
     224    <!-- hidden form for POST submition -->
     225    <form method="post" action="" target="" id="invokeForm" style="display:none;"></form>
     226   
     227  </div> <!-- the_content -->
     228
     229</div> <!-- the_page -->
     230
     231<div id="the_footer">
     232  Copyright © 2002-2012 <a href="http://piwigo.org">Piwigo Team</a>
     233</div> <!-- the_footer -->
     234
     235
     236<script type="text/javascript"> 
     237// global vars
     238var cachedMethods = new Array;
     239var ws_url = "http://";
     240
     241// automatic detection of ws_url
     242match = document.location.toString().match(/^(https?.*\/)tools\/ws\.html?/);
     243if (match==null) {
     244  askForUrl();
     245}
     246else {
     247  ws_url = match[1]+'ws.php';
     248  getMethodList();
     249}
     250
     251// manual set of ws_url
     252$("#urlForm").submit(function() {
     253  ws_url = $(this).children("input[name='ws_url']").val();
     254  getMethodList();
     255  return false;
     256});
     257
     258// invoke buttons
     259$("#invokeMethod").click(function() {
     260  invokeMethod($("#methodName").html(), false);
     261  return false;
     262});
     263$("#invokeMethodBlank").click(function() {
     264  invokeMethod($("#methodName").html(), true);
     265  return false;
     266});
     267
     268// resizable iframe
     269$("#increaseIframe").click(function() {
     270  $("#iframeWrapper").css('height', $("#iframeWrapper").height()+100);
     271  adaptHeight();
     272});
     273$("#decreaseIframe").click(function() {
     274  if ($("#iframeWrapper").height() > 200) {
     275    $("#iframeWrapper").css('height', $("#iframeWrapper").height()-100);
     276    adaptHeight();
     277  }
     278});
     279
     280// mask all wrappers
     281function resetDisplay() {
     282  $("#errorWrapper").hide();
     283  $("#methodWrapper").hide();
     284  $("#methodName").hide();
     285  $("#urlForm").hide();
     286  $("#methodDescription").hide();
     287  $("#invokeFrame").attr('src','');
     288}
     289
     290// give the same size to methods list and main page
     291function adaptHeight()
    11292{
    12   $(id).style.visibility = vis;
    13 }
    14 
    15 function dumpError(err)
    16 {
    17         var s= 'Error ';
    18         if ('string' === typeof err )
    19          s += err;
    20         else
    21         {
    22     s += err.name+'<br/>';
    23          s += err.message;
    24          if (err.stack!=null)
    25          {//mozilla only
    26                 s += '<br/><small><pre>'+ err.stack + '</pre></small>';
    27          }
    28         }
    29   $("error").update(s);
    30 }
    31 
    32 var gServiceUrl;
    33 var gCachedMethods;
    34 
    35 Ajax.Responders.register({
    36 
    37 onException: function(req, err) {
    38         try {
    39                 document.pwgError = err;
    40                 dumpError(err);
    41         }
    42         catch (e)
    43         {
    44                 alert (err);
    45                 alert (err.message);
    46         }
    47 },
    48 
    49 onComplete: function(req, transport) {
    50         if (!req.responseIsSuccess())
    51         {
    52                 var s = 'Completion failure\n' + transport.status + ' ' + transport.statusText;
    53                 if (transport.status>=300)
    54                 {
    55                         s += '\n';
    56                         s += transport.responseText.substr(0,1000);
    57                 }
    58                 dumpError(s);
    59         }
    60  }
    61 }
    62 );
    63 
    64 
    65 function pwgGetJsonResult(transport)
    66 {
    67   var resp;
     293  $("#the_page").css('height', 'auto');
     294  $("#the_methods").css('height', 'auto');
     295 
     296  min_h = $(window).height()-$("#the_header").outerHeight()-$("#the_footer").outerHeight()-3;
     297  h = Math.max(min_h, Math.max($("#the_methods").height(), $("#the_page").height()));
     298 
     299  $("#the_page").css('height', h);
     300  $("#the_methods").css('height', h);
     301}
     302
     303// display error wrapper
     304function displayError(error) {
     305  resetDisplay();
     306  $("#errorWrapper").html("<b>Error:</b> "+ error).show();
     307  adaptHeight();
     308}
     309
     310// display ws_url form
     311function askForUrl() {
     312  if ($("#urlForm input[name='ws_url']").val() == "") {
     313    $("#urlForm input[name='ws_url']").val(ws_url);
     314  }
     315  $("#urlForm").show();
     316  displayError("can't contact web-services, please give absolute url to 'ws.php'");
     317}
     318
     319// parse Piwigo JSON
     320function parsePwgJSON(json) {
    68321  try {
    69                 eval('resp = ' + transport.responseText);
    70         }
    71         catch (e)
    72         {
    73                 var s = e.message;
    74                 s += '\n' + transport.responseText.substr(0,1000).escapeHTML();
    75                 throw new Error( s );
    76         }
    77   if (resp==null || resp.result==null || resp.stat==null || resp.stat!='ok')
    78   {
    79     var s = 'JSON evaluation error';
    80     if (resp)
    81     {
    82                         if (resp.stat!=null) s+= '\n'+resp.stat;
    83       if (resp.message!=null) s+= '\n'+ resp.message;
    84     }
    85     throw new Error(s);
    86   }
     322    resp = jQuery.parseJSON(json);
     323    if (resp==null | resp.result==null | resp.stat==null | resp.stat!='ok') {
     324      throw new Error();
     325    }
     326  }
     327  catch(e) {
     328    displayError("unable to parse JSON string");
     329    resp = {"stat": "ko", "result": "null"};
     330  }
     331 
    87332  return resp.result;
    88333}
    89334
    90 function pwgChangeUrl()
    91 {
    92   $("error").update("");
    93   setVisibility("methodListWrapper", "hidden");
    94   $("methodList").update("");
    95   setVisibility("methodWrapper", "hidden");
    96   setVisibility("methodDetailWrapper", "hidden");
    97 
    98   gServiceUrl = $F('ws_url');
    99   gCachedMethods = new Hash();
    100 
    101   try {
    102                 var ajaxReq = new Ajax.Request(
    103                                 gServiceUrl,
    104                                 {method:'get', parameters:'format=json&method=reflection.getMethodList',
    105                                  onSuccess: function (r) { onSuccess_getMethodList(r); }
    106                                 }
    107                         )
    108   }catch (e)
    109   {
    110     dumpError(e);
    111   }
     335// fetch methods list
     336function getMethodList() {
     337  resetDisplay();
     338 
     339  $.ajax({
     340    type: "GET",
     341    url: ws_url,
     342    data: { format: "json", method: "reflection.getMethodList" }
     343  }).done(function(result) {
     344    result = parsePwgJSON(result);
     345   
     346    if (result!=null) {
     347      methods = result.methods;
     348     
     349      var ml = '';
     350      for (var i=0; i<methods.length; i++)
     351      {
     352        ml += '<li><a href="#">'+ methods[i]+'</a></li>';
     353      }
     354      $("#methodsList").html(ml).show();
     355     
     356      adaptHeight();
     357     
     358      // trigger method selection
     359      $("#methodsList li a").click(function() {
     360        selectMethod($(this).html());
     361        return false;
     362      });
     363    }
     364  }).error(function(jqXHR, textStatus, errorThrown) {
     365    askForUrl();
     366  });
     367}
     368
     369// select method
     370function selectMethod(methodName) {
     371  $("#introMessage").hide();
     372 
     373  if (cachedMethods[ methodName ]) {
     374    fillNewMethod(methodName);
     375  }
     376  else {
     377    $.ajax({
     378      type: "GET",
     379      url: ws_url,
     380      data: { format: "json", method: "reflection.getMethodDetails", methodName: methodName }
     381    }).done(function(result) {
     382      result = parsePwgJSON(result);
     383   
     384      if (result!=null) {
     385        cachedMethods[ methodName ] = result;
     386        fillNewMethod(methodName);
     387      }
     388    }).error(function(jqXHR, textStatus, errorThrown) {
     389      displayError("unknown error");
     390    });
     391  }
     392}
     393
     394// display method details
     395function fillNewMethod(methodName) {
     396  resetDisplay();
     397 
     398  method = cachedMethods[ methodName ];
     399 
     400  $("#methodName").html(method.name).show();
     401 
     402  if (method.description != "") {
     403    $("#methodDescription blockquote").html(method.description);
     404    $("#methodDescription").show();
     405  }
     406 
     407  var methodParams = '';
     408  if (method.params && method.params.length>0) {
     409    for (var i=0; i<method.params.length; i++) {
     410      var isOptional = method.params[i].optional;
     411      var acceptArray = method.params[i].acceptArray;
     412      var defaultValue = method.params[i].defaultValue == null ? '' : method.params[i].defaultValue;
     413     
     414      // if an array is direclty printed, the delimiter is a comma where we use a pipe
     415      if (typeof defaultValue == 'object') {
     416        defaultValue = defaultValue.join('|');
     417      }
     418
     419      methodParams+= '<tr>'+
     420        '<td>'+ method.params[i].name +'</td>'+
     421        '<td class="mini">'+ (isOptional ? '?':'*') + (acceptArray ? ' []':'') +'</td>'+
     422        '<td class="input"><input type="text" class="methodParameterValue" data-id="'+ i +'" value="'+ defaultValue +'"></td>'+
     423        '<td class="mini"><input type="checkbox" class="methodParameterSend" data-id="'+ i +'" '+ (isOptional ? '':'checked="checked"') +'></td>'+
     424      '</tr>';
     425    }
     426        }
     427  else {
     428    methodParams = '<tr><td colspan="4">This method takes no parameters</td></tr>';
     429  }
     430 
     431  $("#methodParams tbody").html(methodParams);
     432  $("#methodWrapper").show();
     433 
     434  adaptHeight();
     435 
     436  // trigger field modification
     437  $("input.methodParameterValue").change(function() {
     438    $("input.methodParameterSend[data-id='"+ $(this).data('id') +"']").attr('checked', 'checked');
     439  });
     440}
     441
     442// invoke method
     443function invokeMethod(methodName, newWindow) {
     444        var method = cachedMethods[ methodName ];
     445
     446  var reqUrl = ws_url +"?format="+ $("#responseFormat").val();
     447 
     448  // GET
     449  if ($("#requestFormat").val() == 'get') {
     450    reqUrl+= "&method="+ methodName;
     451   
     452    for (var i=0; i<method.params.length; i++) {
     453      if (! $("input.methodParameterSend[data-id='"+ i +"']").is(":checked")) {
     454        continue;
     455      }
     456
     457      var paramValue = $("input.methodParameterValue[data-id='"+ i +"']").val();
     458     
     459      var paramSplitted = paramValue.split('|');
     460      if (method.params[i].acceptArray &&  paramSplitted.length > 1) {
     461        $.each(paramSplitted, function(v) {
     462          reqUrl+= '&'+ method.params[i].name +'[]='+ paramSplitted[v];
     463        });
     464      }
     465      else {
     466        reqUrl+= '&'+ method.params[i].name +'='+ paramValue;
     467      }
     468    }
     469   
     470    if (newWindow) {
     471      window.open(reqUrl);
     472    }
     473    else {
     474      $("#invokeFrame").attr('src', reqUrl);
     475    }
     476  }
     477  // POST
     478  else {
     479    var form = $("#invokeForm");
     480    form.attr('action', reqUrl);
     481   
     482    var t = '<input type="hidden" name="method" value="'+ methodName +'">';
     483   
     484    for (var i=0; i<method.params.length; i++) {
     485      if (! $("input.methodParameterSend[data-id='"+ i +"']").is(":checked")) {
     486        continue;
     487      }
     488
     489      var paramValue = $("input.methodParameterValue[data-id='"+ i +"']").val();
     490
     491      var paramSplitted = paramValue.split('|');
     492      if (method.params[i].acceptArray &&  paramSplitted.length > 1) {
     493        $.each(paramSplitted, function(v) {
     494          t+= '<input type="hidden" name="'+ method.params[i].name +'[]" value="'+ paramSplitted[v] +'">';
     495        });
     496      }
     497      else {
     498        t+= '<input type="hidden" name="'+ method.params[i].name +'" value="'+ paramValue +'">';
     499      }
     500    }
     501   
     502    form.html(t);
     503    form.attr('target', newWindow ? "_blank" : "invokeFrame");
     504    form.submit();
     505  }
     506 
    112507  return false;
    113508}
    114 
    115 function onSuccess_getMethodList(transport)
    116 {
    117         var result = pwgGetJsonResult(transport);
    118         var ml = '';
    119         for (var i=0; i<result.methods.length; i++)
    120         {
    121                 ml += '<li><a href="#" onclick="return pwgSelectMethod(this.innerHTML)">'+ result.methods[i]+'</a></li>';
    122         }
    123         $("methodList").update(ml);
    124         setVisibility("methodListWrapper", "visible");
    125 }
    126 
    127 function pwgSelectMethod(methodName)
    128 {
    129   $("error").update("");
    130   $("methodName").update(methodName);
    131   setVisibility("methodDetailWrapper", "hidden");
    132   setVisibility("methodWrapper", "visible");
    133 
    134   if ( gCachedMethods[methodName] )
    135     fillNewMethod( gCachedMethods[methodName] );
    136   else
    137   {
    138     try {
    139                 var ajaxReq = new Ajax.Request(
    140                                 gServiceUrl,
    141                                 {method:'get', parameters:'format=json&method=reflection.getMethodDetails&methodName='+methodName,
    142                                  onSuccess: function (r) { onSuccess_getMethodDetails(r); }
    143                                 }
    144                         )
    145     }catch (e)
    146     {
    147       dumpError( e );
    148     }
    149   }
    150   return false;
    151 }
    152 
    153 function onSuccess_getMethodDetails(transport)
    154 {
    155         var result = pwgGetJsonResult(transport);
    156   fillNewMethod( gCachedMethods[result.name] = result );
    157 }
    158 
    159 function fillNewMethod(method)
    160 {
    161         var methodParamsElt = $("methodParams");
    162         while (methodParamsElt.tBodies[0].rows.length)
    163                 methodParamsElt.tBodies[0].deleteRow(methodParamsElt.tBodies[0].rows.length-1);
    164 
    165         if (method.params && method.params.length>0)
    166         {
    167                 for (var i=0; i<method.params.length; i++)
    168                 {
    169                         var row = methodParamsElt.tBodies[0].insertRow(-1);
    170                         var isOptional = method.params[i].optional;
    171                         var acceptArray = method.params[i].acceptArray;
    172                         var defaultValue = method.params[i].defaultValue == null ? '' : method.params[i].defaultValue;
    173 
    174                         row.insertCell(0).innerHTML = method.params[i].name;
    175                         row.insertCell(1).innerHTML = '<span title="parameter is '+(isOptional ? 'optional':'required') +'">'+(isOptional ? '?':'*')+'</span>'
    176           + (method.params[i].acceptArray ? ' <span title="parameter can be an array; use | (pipe) character to split values">[ ]</span>':'');
    177                         row.insertCell(2).innerHTML = '<input id="methodParameterSend_'+i+'" type="checkbox" '+(isOptional ? '':'checked="checked"')+'/>';
    178                         row.insertCell(3).innerHTML = '<input id="methodParameterValue_'+i+'"" value="'+defaultValue+'" style="width:99%" onchange="$(\'methodParameterSend_'+i+'\').checked=true;"/>';
    179                 }
    180         }
    181         $("methodDescription").update(method.description);
    182         setVisibility("methodDetailWrapper", "visible");
    183 }
    184 
    185 function pwgInvokeMethod( newWindow )
    186 {
    187         var methodName = $('methodName').innerHTML;
    188         var method = gCachedMethods[methodName];
    189 
    190   var reqUrl = gServiceUrl;
    191   reqUrl += "?format="+$F('responseFormat');
    192 
    193   if ($('requestFormat').value == 'get')
    194   {
    195     reqUrl += "&method="+methodName;
    196     for ( var i=0; i<method.params.length; i++)
    197     {
    198       if (! $('methodParameterSend_'+i).checked)
    199         continue;
    200 
    201       if ( method.params[i].acceptArray &&  $F('methodParameterValue_'+i).split('|').length > 1 )
    202       {
    203         $F('methodParameterValue_'+i).split('|').each(
    204             function(v) {
    205               reqUrl += '&'+method.params[i].name+'[]='+v;
    206             }
    207           );
    208       }
    209       else
    210         reqUrl += '&'+method.params[i].name+'='+$F('methodParameterValue_'+i);
    211     }
    212     if ( !newWindow )
    213       $("invokeFrame").src = reqUrl;
    214     else
    215       window.open(reqUrl);
    216   }
    217   else
    218   {
    219     var form = $("invokeForm");
    220     form.action = reqUrl;
    221     var t = '<input type="hidden" name="'+'method'+'" value="'+methodName+'"/>';
    222     for ( var i=0; i<method.params.length; i++)
    223     {
    224       if (! $('methodParameterSend_'+i).checked)
    225         continue;
    226 
    227       if ( method.params[i].acceptArray &&  $F('methodParameterValue_'+i).split('|').length > 1 )
    228       {
    229         $F('methodParameterValue_'+i).split('|').each(
    230             function(v) {
    231               t += '<input type="hidden" name="'+method.params[i].name+'[]" value="'+v+'"/>';
    232             }
    233           );
    234       }
    235       else
    236         t += '<input type="hidden" name="'+method.params[i].name+'" value="'+$F('methodParameterValue_'+i)+'"/>';
    237     }
    238     form.innerHTML = t;
    239     form.target = newWindow ? "_blank" : "invokeFrame";
    240     form.submit();
    241   }
    242   return false;
    243 }
    244509</script>
    245 
    246 
    247 <style>
    248 #methodListWrapper {
    249   width: 13em;
    250   float: left;
    251   display: inline;
    252   visibility: hidden;
    253 }
    254 
    255 #methodList {
    256   padding-left: 10px;
    257   margin-left: 15px;
    258 }
    259 
    260 #methodWrapper {
    261   margin-left: 14em;
    262   visibility: hidden;
    263 }
    264 
    265 #methodName {
    266   margin-top: 0;
    267   margin-bottom: 3px;
    268 }
    269 
    270 
    271 #error {
    272   height: 90px;
    273   overflow: scroll;
    274   color: red;
    275 }
    276 
    277 #methodParams {
    278   border-collapse: collapse;
    279   font-size: small;
    280 }
    281 
    282 #methodParams input {
    283   font-size: 90%;
    284   border: 1px solid black;
    285   text-indent: 2px;
    286 }
    287 
    288 
    289 a {
    290   color: #02f;
    291   background-color: white;
    292   text-decoration: underline;
    293 }
    294 
    295 a:hover {
    296   color: white;
    297   background-color: #02f;
    298   text-decoration: none;
    299   cursor:pointer;
    300 }
    301 
    302 </style>
    303 
    304 </head>
    305 
    306 
    307 <body>
    308 
    309 <div>
    310  <label>PWG Web service url
    311   <input name="ws_url" id="ws_url" size="64"/>
    312 <script type="text/javascript">
    313   var match = document.location.toString().match(/^(https?.*\/)tools\/ws\.html?$/);
    314   if (match!=null) $('ws_url').value = match[1]+'ws.php';
    315 </script>
    316  </label>
    317  <a href="#" onclick="return pwgChangeUrl();">Go!</a>
    318 </div>
    319 
    320 <div id="error">
    321 </div>
    322 
    323 <div>
    324 
    325 <div id="methodListWrapper"><h2>Methods</h2>
    326   <ul id="methodList">
    327     <li><a href="#" onclick="return pwgSelectMethod(this.innerHTML)">getVersion</a></li>
    328   </ul>
    329 </div>
    330 
    331 <div id="methodWrapper">
    332   <h2 id="methodName"></h2>
    333   <div id="methodDetailWrapper">
    334 
    335     <table>
    336     <tr style="vertical-align:top">
    337 
    338     <td>
    339       <div id="methodDescription"></div>
    340       <table>
    341         <tr>
    342           <td>Request format:</td>
    343           <td>
    344             <select id="requestFormat">
    345               <option value="get" selected="selected">GET</option>
    346               <option value="post">POST</option>
    347             </select>
    348           </td>
    349         </tr>
    350 
    351         <tr>
    352           <td>Response format:</td>
    353           <td>
    354             <select id="responseFormat">
    355               <option value="rest" selected="selected">REST (xml)</option>
    356               <option value="json">JSON</option>
    357               <option value="php">PHP serial</option>
    358               <option value="xmlrpc">XML RPC</option>
    359             </select>
    360           </td>
    361         </tr>
    362       </table>
    363       <p>
    364         <a href="#" onclick="return pwgInvokeMethod(false)">Invoke</a>
    365         <a href="#" onclick="return pwgInvokeMethod(true)">Invoke (new Window)</a>
    366       </p>
    367     </td>
    368 
    369 
    370     <td>
    371       <table id="methodParams"  border="1" cellspacing="0" cellpadding="2px">
    372         <thead>
    373           <tr>
    374             <td style="width:150px">Parameter</td>
    375             <td>Extra</td>
    376             <td>Send</td>
    377             <td style="width:160px">Value</td>
    378           </tr>
    379         </thead>
    380         <tbody>
    381         </tbody>
    382       </table>
    383     </td>
    384 
    385     </tr>
    386     </table>
    387 
    388                 <div style="display:none;">
    389                         <!-- hiddenForm for POST -->
    390                         <form method="post" action="" target="invokeFrame" id="invokeForm">
    391                                 <input type="submit" value="submit"/>
    392                         </form>
    393                 </div>
    394 
    395     <iframe width="100%" height="400px" id="invokeFrame" name="invokeFrame" style="clear:both"></iframe>
    396   </div> <!-- methodDetailWrapper -->
    397 </div> <!-- methodWrapper -->
    398 
    399 </div>
    400510
    401511</body>
Note: See TracChangeset for help on using the changeset viewer.