phpMyEdit.class.php 104 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024
  1. <?php
  2. if (! function_exists('__')) {
  3. function __($str, $default=null, $module="GENERAL") {
  4. return $str;
  5. }
  6. }
  7. $opts['display']['time'] = false;
  8. $skipOpts = array();
  9. foreach ($opts['fdd'] as $k => $v) {
  10. if (isset($skipOpts[$k])) continue;
  11. $k2 = $k;
  12. if (ereg('^`.*`$', $k)) $k2 = substr($k, 1, -1);
  13. $tb = $opts['tb'];
  14. if ($opts['tb2']) $tb = $opts['tb2'];
  15. $opts['fdd'][$k]['name'] = __($k2, $v['name'],$tb);
  16. if (isset($opts['fdd'][$k.'x'])) {
  17. $skipOpts[$k.'x'] = 1;
  18. $opts['fdd'][$k.'x']['name'] = $opts['fdd'][$k]['name'];
  19. }
  20. }
  21. /*
  22. * phpMyEdit - instant MySQL table editor and code generator
  23. *
  24. * phpMyEdit.class.php - main table editor class definition file
  25. * ____________________________________________________________
  26. *
  27. * Copyright (c) 1999-2002 John McCreesh <jpmcc@users.sourceforge.net>
  28. * Copyright (c) 2001-2002 Jim Kraai <jkraai@users.sourceforge.net>
  29. * Versions 5.0 and higher developed by Ondrej Jombik <nepto@php.net>
  30. * Copyright (c) 2002-2004 Platon SDG, http://platon.sk/
  31. * All rights reserved.
  32. *
  33. * See README file for more information about this software.
  34. * See COPYING file for license information.
  35. *
  36. * Download the latest version from
  37. * http://platon.sk/projects/phpMyEdit/
  38. */
  39. /* $Platon: phpMyEdit/phpMyEdit.class.php,v 1.121 2004/01/26 17:17:49 nepto Exp $ */
  40. /* This is a generic table editing program. The table and fields to be
  41. edited are defined in the calling program.
  42. This program works in three passes.
  43. * Pass 1 (the last part of the program) displays the selected MySQL
  44. table in a scrolling table on the screen. Radio buttons are used to
  45. select a record for editing or deletion. If the user chooses Add,
  46. Change, Copy, View or Delete buttons.
  47. * Pass 2 starts, displaying the selected record. If the user chooses
  48. the Save button from this screen.
  49. * Pass 3 processes the update and the display returns to the
  50. original table view (Pass 1).
  51. */
  52. class phpMyEdit_timer /* {{{ */
  53. {
  54. var $startTime;
  55. var $started;
  56. var $insert_id;
  57. function phpMyEdit_timer($start = true)
  58. {
  59. $this->started = false;
  60. if ($start) {
  61. $this->start();
  62. }
  63. }
  64. function start()
  65. {
  66. $startMtime = explode(' ', microtime());
  67. $this->startTime = (double) $startMtime[0] + (double) $startMtime[1];
  68. $this->started = true;
  69. }
  70. function end($iterations = 1)
  71. {
  72. // get the time, check whether the timer was started later
  73. $endMtime = explode(' ', microtime());
  74. if ($this->started) {
  75. $endTime = (double)($endMtime[0])+(double)($endMtime[1]);
  76. $dur = $endTime - $this->startTime;
  77. $avg = 1000 * $dur / $iterations;
  78. $avg = round(1000 * $avg) / 1000;
  79. return $avg;
  80. } else {
  81. return 'phpMyEdit_timer ERROR: timer not started';
  82. }
  83. }
  84. } /* }}} */
  85. if (! function_exists('array_search')) { /* {{{ */
  86. function array_search($needle, $haystack)
  87. {
  88. foreach ($haystack as $key => $value) {
  89. if ($needle == $value)
  90. return $key;
  91. }
  92. return false;
  93. }
  94. } /* }}} */
  95. if (! function_exists('realpath')) { /* {{{ */
  96. function realpath($path)
  97. {
  98. return $path;
  99. }
  100. } /* }}} */
  101. class phpMyEdit
  102. {
  103. // Class variables {{{
  104. // Database handling
  105. var $hn; // hostname
  106. var $un; // user name
  107. var $pw; // password
  108. var $db; // database
  109. var $tb; // table
  110. var $dbh; // database handle
  111. // Record manipulation
  112. var $key; // name of field which is the unique key
  113. var $key_num; // number of field which is the unique key
  114. var $key_type; // type of key field (int/real/string/date etc.)
  115. var $key_delim; // character used for key value quoting
  116. var $rec; // number of record selected for editing
  117. var $inc; // number of records to display
  118. var $fm; // first record to display
  119. var $fl; // is the filter row displayed (boolean)
  120. var $fds; // sql field names
  121. var $num_fds; // number of fields
  122. var $options; // options for users: ACDFVPI
  123. var $fdd; // field definitions
  124. var $qfn; // value of all filters used during the last pass
  125. var $sfn; // sort field number (- = descending sort order)
  126. // Operation
  127. var $navop; // navigation buttons/operations
  128. var $sw; // filter display/hide/clear button
  129. var $operation; // operation to do: Add, Change, Delete
  130. var $saveadd;
  131. var $moreadd;
  132. var $savechange;
  133. var $savedelete;
  134. var $canceladd;
  135. var $cancelview;
  136. var $cancelchange;
  137. var $canceldelete;
  138. // Additional features
  139. var $labels; // multilingual labels
  140. var $cgi; // CGI variable features array
  141. var $url; // URL array
  142. var $message; // informational message to print
  143. var $notify; // change notification e-mail adresses
  144. var $logtable; // name of optional logtable
  145. var $navigation; // navigation style
  146. var $tabs; // TAB names
  147. var $timer = null; // phpMyEdit_timer object
  148. // Predefined variables
  149. var $comp_ops = array('<'=>'<','<='=>'<=','='=>'=','>='=>'>=','>'=>'>');
  150. var $sql_aggrs = array(
  151. 'sum' => 'Total',
  152. 'avg' => 'Average',
  153. 'min' => 'Minimum',
  154. 'max' => 'Maximum',
  155. 'count' => 'Count');
  156. var $page_types = array(
  157. 'L' => 'list',
  158. 'F' => 'filter',
  159. 'A' => 'add',
  160. 'V' => 'view',
  161. 'C' => 'change',
  162. 'P' => 'copy',
  163. 'D' => 'delete'
  164. );
  165. // }}}
  166. /*
  167. * column specific functions
  168. */
  169. function col_has_sql($k) { return isset($this->fdd[$k]['sql']); }
  170. function col_has_sqlw($k) { return isset($this->fdd[$k]['sqlw']) && !$this->virtual($k); }
  171. function col_has_values($k) { return isset($this->fdd[$k]['values']) || isset($this->fdd[$k]['values2']); }
  172. function col_has_URL($k) { return isset($this->fdd[$k]['URL'])
  173. || isset($this->fdd[$k]['URLprefix']) || isset($this->fdd[$k]['URLpostfix']); }
  174. function col_has_multiple_select($k)
  175. { return $this->fdd[$k]['select'] == 'M' && ! $this->fdd[$k]['values']['table']; }
  176. function col_has_datemask($k)
  177. { return isset($this->fdd[$k]['datemask']) || isset($this->fdd[$k]['strftimemask']); }
  178. /*
  179. * functions for indicating whether navigation style is enabled
  180. */
  181. function nav_buttons() { return stristr($this->navigation, 'B'); }
  182. function nav_text_links() { return stristr($this->navigation, 'T'); }
  183. function nav_graphic_links() { return stristr($this->navigation, 'G'); }
  184. function nav_up() { return stristr($this->navigation, 'U'); }
  185. function nav_down() { return stristr($this->navigation, 'D'); }
  186. /*
  187. * functions for indicating whether operations are enabled
  188. */
  189. function initial_sort_suppressed() { return (stristr ($this->options, 'I')); }
  190. function add_enabled() { return stristr($this->options, 'A'); }
  191. function change_enabled() { return stristr($this->options, 'C'); }
  192. function delete_enabled($key_rec=null) {
  193. if (!is_null($key_rec)) $key_rec = urldecode($key_rec);
  194. if (is_null($key_rec) && $_GET['rec']) $key_rec = $_GET['rec'];
  195. if (is_null($key_rec) && $_POST['rec']) $key_rec = $_POST['rec'];
  196. if (function_exists('flag_enabled')) if (flag_enabled($key_rec) !== 1) return false;
  197. return stristr($this->options, 'D');
  198. }
  199. function filter_enabled() { return stristr($this->options, 'F'); }
  200. function view_enabled() { return stristr($this->options, 'V'); }
  201. function copy_enabled() { return stristr($this->options, 'P') && $this->add_enabled(); }
  202. function tabs_enabled() { return $this->display['tabs'] && count($this->tabs) > 0; }
  203. function hidden($k) { return stristr($this->fdd[$k]['input'],'H') || stristr($this->fdd[$k]['options'],'H'); }
  204. function password($k) { return stristr($this->fdd[$k]['input'],'W') || stristr($this->fdd[$k]['options'],'W'); }
  205. function readonly($k) { return stristr($this->fdd[$k]['input'],'R') || stristr($this->fdd[$k]['options'],'R') || $this->virtual($k); }
  206. function virtual($k) { return stristr($this->fdd[$k]['input'],'V') && $this->col_has_sql($k); }
  207. function add_operation() { return $this->operation == $this->labels['Add'] && $this->add_enabled(); }
  208. function change_operation() { return $this->operation == $this->labels['Change'] && $this->change_enabled(); }
  209. function copy_operation() { return $this->operation == $this->labels['Copy'] && $this->copy_enabled(); }
  210. function delete_operation() { return $this->operation == $this->labels['Delete'] && $this->delete_enabled(); }
  211. function view_operation() { return $this->operation == $this->labels['View'] && $this->view_enabled(); }
  212. function filter_operation() { return $this->fl && $this->filter_enabled() && $this->list_operation(); }
  213. function list_operation() { /* covers also filtering page */ return ! $this->change_operation()
  214. && ! $this->add_operation() && ! $this->copy_operation()
  215. && ! $this->delete_operation() && ! $this->view_operation(); }
  216. function next_operation() { return $this->navop == $this->labels['Next']; }
  217. function prev_operation() { return $this->navop == $this->labels['Prev']; }
  218. function first_operation() { return $this->navop == $this->labels['First']; }
  219. function last_operation() { return $this->navop == $this->labels['Last']; }
  220. function goto_operation() { return $this->navop == $this->labels['Go to']; }
  221. function clear_operation() { return $this->sw == $this->labels['Clear']; }
  222. function add_canceled() { return $this->canceladd == $this->labels['Cancel']; }
  223. function view_canceled() { return $this->cancelview == $this->labels['Cancel']; }
  224. function change_canceled() { return $this->cancelchange == $this->labels['Cancel']; }
  225. function delete_canceled() { return $this->canceldelete == $this->labels['Cancel']; }
  226. function is_values2($k, $val = 'X') /* {{{ */
  227. {
  228. return $val === null ||
  229. (isset($this->fdd[$k]['values2']) && !isset($this->fdd[$k]['values']['table']));
  230. } /* }}} */
  231. function processed($k) /* {{{ */
  232. {
  233. if ($this->virtual($k)) {
  234. return false;
  235. }
  236. $options = @$this->fdd[$k]['options'];
  237. if (! isset($options)) {
  238. return true;
  239. }
  240. return
  241. ($this->saveadd == $this->labels['Save'] && stristr($options, 'A')) ||
  242. ($this->moreadd == $this->labels['More'] && stristr($options, 'A')) ||
  243. ($this->savechange == $this->labels['Save'] && stristr($options, 'C')) ||
  244. ($this->morechange == $this->labels['Apply'] && stristr($options, 'C')) ||
  245. ($this->savechange == $this->labels['Save'] && stristr($options, 'P')) ||
  246. ($this->savedelete == $this->labels['Save'] && stristr($options, 'D'));
  247. } /* }}} */
  248. function displayed($k) /* {{{ */
  249. {
  250. if (is_numeric($k)) {
  251. $k = $this->fds[$k];
  252. }
  253. $options = @$this->fdd[$k]['options'];
  254. if (! isset($options)) {
  255. return true;
  256. }
  257. return
  258. ($this->add_operation() && stristr($options, 'A')) ||
  259. ($this->view_operation() && stristr($options, 'V')) ||
  260. ($this->change_operation() && stristr($options, 'C')) ||
  261. ($this->copy_operation() && stristr($options, 'P')) ||
  262. ($this->delete_operation() && stristr($options, 'D')) ||
  263. ($this->filter_operation() && stristr($options, 'F')) ||
  264. ($this->list_operation() && stristr($options, 'L'));
  265. } /* }}} */
  266. function debug_var($name, $val) /* {{{ */
  267. {
  268. if (is_array($val) || is_object($val)) {
  269. echo "<pre>$name\n";
  270. ob_start();
  271. //print_r($val);
  272. var_dump($val);
  273. $content = ob_get_contents();
  274. ob_end_clean();
  275. echo htmlspecialchars($content);
  276. echo "</pre>\n";
  277. } else {
  278. echo 'debug_var()::<i>',htmlspecialchars($name),'</i>';
  279. echo '::<b>',htmlspecialchars($val),'</b>::',"<br>\n";
  280. }
  281. } /* }}} */
  282. function myquery($qry, $line = 0, $debug = 0) /* {{{ */
  283. {
  284. global $debug_query;
  285. if ($debug_query || $debug) {
  286. $line = intval($line);
  287. echo '<h4>MySQL query at line ',$line,'</h4>',htmlspecialchars($qry),'<hr>',"\n";
  288. }
  289. $ret = @mysql_db_query($this->db, $qry, $this->dbh);
  290. if (! $ret) {
  291. echo '<font color=red>';
  292. echo '<h4>MySQL error ',mysql_errno($this->dbh),'</h4>';
  293. echo htmlspecialchars(mysql_error($this->dbh)),'<hr>',"\n";
  294. echo '</font>';
  295. }
  296. return $ret;
  297. } /* }}} */
  298. function make_language_labels($language) /* {{{ */
  299. {
  300. // just try the first language and variant
  301. // this isn't content-negotiation rfc compliant
  302. $language = strtoupper(substr($language,0,5));
  303. // try the full language w/ variant
  304. $file = $this->dir['lang'].'PME.lang.'.$language.'.inc';
  305. if (false && ! file_exists($file)) {
  306. // try the language w/o variant
  307. $file = $this->dir['lang'].'PME.lang.'.substr($language,0,2).'.inc';
  308. }
  309. if (true || ! file_exists($file)) {
  310. // default to classical English
  311. $file = $this->dir['lang'].'PME.lang.EN.inc';
  312. }
  313. $ret = @include($file);
  314. if (! is_array($ret)) {
  315. return $ret;
  316. }
  317. $small = array(
  318. 'Search' => 'v',
  319. 'Hide' => '^',
  320. 'Clear' => 'X',
  321. 'Query' => htmlspecialchars('>'));
  322. if ((!$this->nav_text_links() && !$this->nav_graphic_links())
  323. || !isset($ret['Search']) || !isset($ret['Query'])
  324. || !isset($ret['Hide']) || !isset($ret['Clear'])) {
  325. foreach ($small as $key => $val) {
  326. $ret[$key] = $val;
  327. }
  328. }
  329. return $ret;
  330. } /* }}} */
  331. function set_values($field_num, $prepend = null, $append = null, $strict = false) /* {{{ */
  332. {
  333. return (array) $prepend + (array) $this->fdd[$field_num]['values2']
  334. + (isset($this->fdd[$field_num]['values']['table']) || $strict
  335. ? $this->set_values_from_table($field_num, $strict)
  336. : array())
  337. + (array) $append;
  338. } /* }}} */
  339. function set_values_from_table($field_num, $strict = false) /* {{{ */
  340. {
  341. $db = &$this->fdd[$field_num]['values']['db'];
  342. $table = &$this->fdd[$field_num]['values']['table'];
  343. $key = &$this->fdd[$field_num]['values']['column'];
  344. $desc = &$this->fdd[$field_num]['values']['description'];
  345. isset($db) || $db = $this->db;
  346. $qparts['type'] = 'select';
  347. if ($table) {
  348. $qparts['select'] = 'DISTINCT '.$table.'.'.$key;
  349. if ($desc && is_array($desc) && is_array($desc['columns'])) {
  350. $qparts['select'] .= ',CONCAT('; // )
  351. $num_cols = sizeof($desc['columns']);
  352. if (isset($desc['divs'][-1])) {
  353. $qparts['select'] .= '"'.addslashes($desc['divs'][-1]).'",';
  354. }
  355. foreach ($desc['columns'] as $key => $val) {
  356. if ($val) {
  357. $qparts['select'] .= $val;
  358. if ($desc['divs'][$key]) {
  359. $qparts['select'] .= ',"'.addslashes($desc['divs'][$key]).'"';
  360. }
  361. $qparts['select'] .= ',';
  362. }
  363. }
  364. $qparts['select']{strlen($qparts['select']) - 1} = ')';
  365. $qparts['select'] .= ' AS PMEalias'.$field_num;
  366. $qparts['orderby'] = empty($desc['orderby'])
  367. ? 'PMEalias'.$field_num : $desc['orderby'];
  368. } else if ($desc && is_array($desc)) {
  369. // TODO
  370. } else if ($desc) {
  371. $qparts['select'] .= ','.$table.'.'.$desc;
  372. $qparts['orderby'] = $desc;
  373. } else if ($key) {
  374. $qparts['orderby'] = $key;
  375. }
  376. //$qparts['from'] = "$db.$table.$sel;
  377. $qparts['from'] = "$db.$table";
  378. $qparts['where'] = $this->fdd[$field_num]['values']['filters'];
  379. if ($this->fdd[$field_num]['values']['orderby']) {
  380. $qparts['orderby'] = $this->fdd[$field_num]['values']['orderby'];
  381. }
  382. } else { /* simple value extraction */
  383. $key = &$this->fds[$field_num];
  384. $this->virtual($field_num) && $key = $this->fqn($field_num);
  385. $qparts['select'] = 'DISTINCT '.$key.' AS PMEkey';
  386. $qparts['orderby'] = 'PMEkey';
  387. $qparts['from'] = $this->db.'.'.$this->tb;
  388. }
  389. $values = array();
  390. $res = $this->myquery($this->query_make($qparts), __LINE__);
  391. while ($row = @mysql_fetch_array($res, MYSQL_NUM)) {
  392. $values[$row[0]] = $desc ? $row[1] : $row[0];
  393. }
  394. return $values;
  395. } /* }}} */
  396. function fqn($field, $dont_desc = false, $dont_cols = false) /* {{{ */
  397. {
  398. is_numeric($field) || $field = array_search($field, $this->fds);
  399. // if read SQL expression exists use it
  400. if ($this->col_has_sql($field))
  401. return $this->fdd[$field]['sql'];
  402. // on copy/change always use simple key retrieving
  403. if ($this->add_operation()
  404. || $this->copy_operation()
  405. || $this->change_operation()) {
  406. $ret = 'PMEtable0.'.$this->fds[$field];
  407. } else {
  408. if ($this->fdd[$this->fds[$field]]['values']['description'] && ! $dont_desc) {
  409. $desc = &$this->fdd[$this->fds[$field]]['values']['description'];
  410. if (is_array($desc) && is_array($desc['columns'])) {
  411. $ret = 'CONCAT('; // )
  412. $num_cols = sizeof($desc['columns']);
  413. if (isset($desc['divs'][-1])) {
  414. $ret .= '"'.addslashes($desc['divs'][-1]).'",';
  415. }
  416. foreach ($desc['columns'] as $key => $val) {
  417. if ($val) {
  418. $ret .= 'PMEjoin'.$field.'.'.$val;
  419. if ($desc['divs'][$key]) {
  420. $ret .= ',"'.addslashes($desc['divs'][$key]).'"';
  421. }
  422. $ret .= ',';
  423. }
  424. }
  425. $ret{strlen($ret) - 1} = ')';
  426. } else if (is_array($desc)) {
  427. // TODO
  428. } else {
  429. $ret = 'PMEjoin'.$field.'.'.$this->fdd[$this->fds[$field]]['values']['description'];
  430. }
  431. // TODO: remove me
  432. } elseif (0 && $this->fdd[$this->fds[$field]]['values']['column'] && ! $dont_cols) {
  433. $ret = 'PMEjoin'.$field.'.'.$this->fdd[$this->fds[$field]]['values']['column'];
  434. } else {
  435. $ret = 'PMEtable0.'.$this->fds[$field];
  436. }
  437. // TODO: not neccessary, remove me!
  438. if (is_array($this->fdd[$this->fds[$field]]['values2'])) {
  439. }
  440. }
  441. return $ret;
  442. } /* }}} */
  443. function create_column_list() /* {{{ */
  444. {
  445. $fields = array();
  446. for ($k = 0; $k < $this->num_fds; $k++) {
  447. if (! $this->displayed[$k] && $k != $this->key_num) {
  448. continue;
  449. }
  450. $fields[] = $this->fqn($k).' AS qf'.$k;
  451. if ($this->col_has_values($k)) {
  452. $fields[] = $this->fqn($k, true, true).' AS qf'.$k.'_idx';
  453. }
  454. if ($this->col_has_datemask($k)) {
  455. $fields[] = 'UNIX_TIMESTAMP('.$this->fqn($k).') AS qf'.$k.'_timestamp';
  456. }
  457. }
  458. return join(',', $fields);
  459. } /* }}} */
  460. function query_make($parts) /* {{{ */
  461. {
  462. foreach ($parts as $k => $v) {
  463. $parts[$k] = trim($parts[$k]);
  464. }
  465. switch ($parts['type']) {
  466. case 'select':
  467. $ret = 'SELECT ';
  468. if ($parts['DISTINCT'])
  469. $ret .= 'DISTINCT ';
  470. $ret .= $parts['select'];
  471. $ret .= ' FROM '.$parts['from'];
  472. if ($parts['where'] != '')
  473. $ret .= ' WHERE '.$parts['where'];
  474. if ($parts['groupby'] != '')
  475. $ret .= ' GROUP BY '.$parts['groupby'];
  476. if ($parts['having'] != '')
  477. $ret .= ' HAVING '.$parts['having'];
  478. if ($parts['orderby'] != '')
  479. $ret .= ' ORDER BY '.$parts['orderby'];
  480. if ($parts['limit'] != '')
  481. $ret .= ' LIMIT '.$parts['limit'];
  482. if ($parts['procedure'] != '')
  483. $ret .= ' PROCEDURE '.$parts['procedure'];
  484. break;
  485. case 'update':
  486. $ret = 'UPDATE '.$parts['table'];
  487. $ret .= ' SET '.$parts['fields'];
  488. if ($parts['where'] != '')
  489. $ret .= ' WHERE '.$parts['where'];
  490. break;
  491. case 'insert':
  492. $ret = 'INSERT INTO '.$parts['table'];
  493. $ret .= ' VALUES '.$parts['values'];
  494. break;
  495. case 'delete':
  496. $ret = 'DELETE FROM '.$parts['table'];
  497. if ($parts['where'] != '')
  498. $ret .= ' WHERE '.$parts['where'];
  499. break;
  500. default:
  501. die('unknown query type');
  502. break;
  503. }
  504. return $ret;
  505. } /* }}} */
  506. function create_join_clause() /* {{{ */
  507. {
  508. $tbs[] = $this->tb;
  509. $join = $this->tb.' AS PMEtable0';
  510. for ($k = 0,$numfds = sizeof($this->fds); $k<$numfds; $k++) {
  511. $field = $this->fds[$k];
  512. if($this->fdd[$field]['values']['db']) {
  513. $db = $this->fdd[$field]['values']['db'];
  514. } else {
  515. $db = $this->db;
  516. }
  517. $table = $this->fdd[$field]['values']['table'];
  518. $id = $this->fdd[$field]['values']['column'];
  519. $desc = $this->fdd[$field]['values']['description'];
  520. if ($desc != '' && $id != '') {
  521. $alias = 'PMEjoin'.$k;
  522. if (!in_array($alias,$tbs)) {
  523. $join .= " LEFT OUTER JOIN $db.$table AS $alias";
  524. $join .= " ON $alias.$id = PMEtable0.$field";
  525. $tbs[]=$alias;
  526. }
  527. }
  528. }
  529. return $join;
  530. } /* }}} */
  531. function make_where_from_query_opts($qp = null, $text = 0) /* {{{ */
  532. {
  533. if ($qp == null) {
  534. $qp = $this->query_opts;
  535. }
  536. $where = array();
  537. foreach ($qp as $field => $ov) {
  538. if (is_numeric($field)) {
  539. $tmp_where = array();
  540. foreach ($ov as $field2 => $ov2) {
  541. $tmp_where[] = sprintf('%s %s %s', $field2, $ov2['oper'], $ov2['value']);
  542. }
  543. $where[] = '('.join(' OR ', $tmp_where).')';
  544. } else {
  545. if (is_array($ov['value'])) {
  546. $tmp_ov_val = '';
  547. foreach ($ov['value'] as $ov_val) {
  548. strlen($tmp_ov_val) > 0 && $tmp_ov_val .= ' OR ';
  549. $tmp_ov_val .= sprintf('FIND_IN_SET("%s",%s)', $ov_val, $field);
  550. }
  551. $where[] = "($tmp_ov_val)";
  552. } else {
  553. $where[] = sprintf('%s %s %s', $field, $ov['oper'], $ov['value']);
  554. }
  555. }
  556. }
  557. // Add any coder specified filters
  558. if (! $text && $this->filters) {
  559. $where[] = '('.$this->filters.')';
  560. }
  561. if (count($where) > 0) {
  562. if ($text) {
  563. return str_replace('%', '*', join(' AND ',$where));
  564. } else {
  565. return join(' AND ',$where);
  566. }
  567. }
  568. return false;
  569. } /* }}} */
  570. function gather_query_opts() /* {{{ */
  571. {
  572. $this->query_opts = array();
  573. $this->prev_qfn = $this->qfn;
  574. $this->qfn = '';
  575. if ($this->clear_operation()) {
  576. return;
  577. }
  578. // gathers query options into an array, $this->query_opts
  579. $qo = array();
  580. for ($k = 0; $k < $this->num_fds; $k++) {
  581. $l = 'qf'.$k;
  582. $lc = 'qf'.$k.'_comp';
  583. $li = 'qf'.$k.'_id';
  584. $m = $this->get_cgi_var($l);
  585. $mc = $this->get_cgi_var($lc);
  586. $mi = $this->get_cgi_var($li);
  587. if (! isset($m) && ! isset($mi)) {
  588. continue;
  589. }
  590. if (is_array($m) || is_array($mi)) {
  591. if (is_array($mi)) {
  592. $m = $mi;
  593. $l = $li;
  594. }
  595. if (in_array('*', $m)) {
  596. continue;
  597. }
  598. if ($this->col_has_values($k) && $this->col_has_multiple_select($k)) {
  599. foreach (array_keys($m) as $key) {
  600. $m[$key] = addslashes($m[$key]);
  601. }
  602. $qo[$this->fqn($k)] = array('value' => $m);
  603. } else {
  604. $qf_op = '';
  605. foreach (array_keys($m) as $key) {
  606. if ($qf_op == '') {
  607. $qf_op = 'IN';
  608. $qf_val = '"'.addslashes($m[$key]).'"';
  609. $afilter = ' IN ("'.addslashes($m[$key]).'"'; // )
  610. } else {
  611. $afilter = $afilter.',"'.addslashes($m[$key]).'"';
  612. $qf_val .= ',"'.addslashes($m[$key]).'"';
  613. }
  614. $this->qfn .= '&'.$l.'['.rawurlencode($key).']='.rawurlencode($m[$key]);
  615. }
  616. $afilter = $afilter.')';
  617. // XXX: $dont_desc and $dont_cols hack
  618. $dont_desc = isset($this->fdd[$k]['values']['description']);
  619. $dont_cols = isset($this->fdd[$k]['values']['column']);
  620. $qo[$this->fqn($k, $dont_desc, $dont_cols)] =
  621. array('oper' => $qf_op, 'value' => "($qf_val)"); // )
  622. }
  623. } else if (isset($mi)) {
  624. if ($mi == '*') {
  625. continue;
  626. }
  627. if ($this->fdd[$k]['select'] != 'M' && $this->fdd[$k]['select'] != 'D' && $mi == '') {
  628. continue;
  629. }
  630. $afilter = addslashes($mi);
  631. $qo[$this->fqn($k, true, true)] = array('oper' => '=', 'value' => "'$afilter'");
  632. $this->qfn .= '&'.$li.'='.rawurlencode($mi);
  633. } else if (isset($m)) {
  634. if ($m == '*') {
  635. continue;
  636. }
  637. if ($this->fdd[$k]['select'] != 'M' && $this->fdd[$k]['select'] != 'D' && $m == '') {
  638. continue;
  639. }
  640. $afilter = addslashes($m);
  641. if ($this->fdd[$k]['select'] == 'N') {
  642. $mc = in_array($mc, $this->comp_ops) ? $mc : '=';
  643. $qo[$this->fqn($k)] = array('oper' => $mc, 'value' => "'$afilter'");
  644. $this->qfn .= '&'.$l .'='.rawurlencode($m);
  645. $this->qfn .= '&'.$lc.'='.rawurlencode($mc);
  646. } else {
  647. $afilter = '%'.str_replace('*', '%', $afilter).'%';
  648. $ids = array();
  649. $ar = array();
  650. $ar[$this->fqn($k)] = array('oper' => 'LIKE', 'value' => "'$afilter'");
  651. if (is_array($this->fdd[$k]['values2'])) {
  652. foreach ($this->fdd[$k]['values2'] as $key => $val) {
  653. if (strlen($m) > 0 && stristr($val, $m)) {
  654. $ids[] = '"'.addslashes($key).'"';
  655. }
  656. }
  657. if (count($ids) > 0) {
  658. $ar[$this->fqn($k, true, true)]
  659. = array('oper' => 'IN', 'value' => '('.join(',', $ids).')');
  660. }
  661. }
  662. $qo[] = $ar;
  663. $this->qfn .= '&'.$l.'='.rawurlencode($m);
  664. }
  665. }
  666. }
  667. $this->query_opts = $qo;
  668. } /* }}} */
  669. /*
  670. * Create JavaScripts
  671. */
  672. function form_begin() /* {{{ */
  673. {
  674. /*
  675. Need a lot of work in here
  676. using something like:
  677. $fdd['fieldname']['validate']['js_regex']='/something/';
  678. $fdd['fieldname']['validate']['php_regex']='something';
  679. */
  680. $page_name = htmlspecialchars($this->page_name);
  681. if ($this->add_operation() || $this->change_operation() || $this->copy_operation()
  682. || $this->view_operation() || $this->delete_operation()) {
  683. $field_to_tab = '';
  684. for ($tab = 0, $k = 0; $k < $this->num_fds; $k++) {
  685. if (isset($this->fdd[$k]['tab'])) {
  686. if ($tab == 0 && $k > 0) {
  687. $this->tabs[0] = 'Initial TAB';
  688. $tab++;
  689. }
  690. $this->tabs[$tab] = @$this->fdd[$k]['tab'];
  691. $tab++;
  692. }
  693. $field_to_tab .= max(0, $tab - 1).', ';
  694. }
  695. if ($this->tabs_enabled()) {
  696. // initial TAB styles
  697. echo '<style type="text/css" media="screen">',"\n";
  698. echo ' #phpMyEdit_tab0 { display: block; }',"\n";
  699. for ($i = 1; $i < count($this->tabs); $i++) {
  700. echo ' #phpMyEdit_tab',$i,' { display: none; }',"\n";
  701. }
  702. echo '</style>',"\n";
  703. // TAB javascripts
  704. echo '<script type="text/javascript"><!--',"\n\n";
  705. echo 'var phpMyEdit_field_to_tab = [',$field_to_tab,'-1];',"\n";
  706. $css_class_name1 = $this->getCSSclass('tab', $position);
  707. $css_class_name2 = $this->getCSSclass('tab-selected', $position);
  708. echo 'var phpMyEdit_current_tab = "phpMyEdit_tab0";
  709. function phpMyEdit_show_tab(tab_name)
  710. {';
  711. if ($this->nav_up()) {
  712. echo '
  713. document.getElementById(phpMyEdit_current_tab+"_up_label").className = "',$css_class_name1,'";
  714. document.getElementById(phpMyEdit_current_tab+"_up_link").className = "',$css_class_name1,'";
  715. document.getElementById(tab_name+"_up_label").className = "',$css_class_name2,'";
  716. document.getElementById(tab_name+"_up_link").className = "',$css_class_name2,'";';
  717. }
  718. if ($this->nav_down()) {
  719. echo '
  720. document.getElementById(phpMyEdit_current_tab+"_down_label").className = "',$css_class_name1,'";
  721. document.getElementById(phpMyEdit_current_tab+"_down_link").className = "',$css_class_name1,'";
  722. document.getElementById(tab_name+"_down_label").className = "',$css_class_name2,'";
  723. document.getElementById(tab_name+"_down_link").className = "',$css_class_name2,'";';
  724. }
  725. echo '
  726. document.getElementById(phpMyEdit_current_tab).style.display = "none";
  727. document.getElementById(tab_name).style.display = "block";
  728. phpMyEdit_current_tab = tab_name;
  729. }',"\n\n";
  730. echo '// --></script>', "\n";
  731. }
  732. }
  733. echo '<script type="text/javascript"><!--',"\n";
  734. echo '
  735. function phpMyEdit_trim(str)
  736. {
  737. while (str.substring(0, 1) == " "
  738. || str.substring(0, 1) == "\\n"
  739. || str.substring(0, 1) == "\\r")
  740. {
  741. str = str.substring(1, str.length);
  742. }
  743. while (str.substring(str.length - 1, str.length) == " "
  744. || str.substring(str.length - 1, str.length) == "\\n"
  745. || str.substring(str.length - 1, str.length) == "\\r")
  746. {
  747. str = str.substring(0, str.length - 1);
  748. }
  749. return str;
  750. }
  751. function phpMyEdit_form_control(theForm)
  752. {',"\n";
  753. if ($this->add_operation() || $this->change_operation() || $this->copy_operation()) {
  754. $required_ar = array();
  755. for ($k = 0; $k < $this->num_fds; $k++) {
  756. if ($this->displayed[$k] && $this->fdd[$k]['required']
  757. && ! $this->readonly($k) && ! $this->hidden($k)) {
  758. $required_ar[] = $k;
  759. if (isset($this->fdd[$k]['regex']['js'])) {
  760. /* TODO: Use a javascript regex to validate it */
  761. }
  762. }
  763. }
  764. if (count($required_ar) > 0) {
  765. foreach ($required_ar as $field_num) {
  766. if ($this->col_has_values($field_num)) {
  767. $condition = 'theForm.%s.selectedIndex == -1';
  768. $multiple = $this->col_has_multiple_select($field_num);
  769. } else {
  770. $condition = 'phpMyEdit_trim(theForm.%s.value) == ""';
  771. $multiple = false;
  772. }
  773. /* Multiple selects have their name like ``name[]''.
  774. It is not possible to work with them directly, because
  775. theForm.name[].something will result into JavaScript
  776. syntax error. Following search algorithm is provided
  777. as a workaround for this.
  778. */
  779. if ($multiple) {
  780. echo '
  781. multiple_select = null;
  782. for (i = 0; i < theForm.length; i++) {
  783. if (theForm.elements[i].name == "',$this->fds[$field_num],'[]") {
  784. multiple_select = theForm.elements[i];
  785. break;
  786. }
  787. }
  788. if (multiple_select != null && multiple_select.selectedIndex == -1) {
  789. alert("',$this->labels['Please enter'],' ',$this->fdd[$field_num]['name'],'.");';
  790. if ($this->tabs_enabled()) {
  791. echo '
  792. phpMyEdit_show_tab("phpMyEdit_tab"+phpMyEdit_field_to_tab['.$field_num.']);';
  793. }
  794. echo '
  795. return false;
  796. }',"\n";
  797. } else {
  798. echo '
  799. if (',sprintf($condition, $this->fds[$field_num]),') {
  800. alert("',$this->labels['Please enter'],' ',$this->fdd[$field_num]['name'],'.");';
  801. if ($this->tabs_enabled()) {
  802. echo '
  803. phpMyEdit_show_tab("phpMyEdit_tab"+phpMyEdit_field_to_tab['.$field_num.']);';
  804. }
  805. echo '
  806. theForm.',$this->fds[$field_num],'.focus();
  807. return false;
  808. }',"\n";
  809. }
  810. }
  811. }
  812. }
  813. echo '
  814. return true;
  815. }',"\n\n";
  816. echo '// --></script>', "\n";
  817. if ($this->filter_operation()) {
  818. echo '<script type="text/javascript"><!--',"\n";
  819. echo '
  820. function phpMyEdit_filter_handler(theForm, theEvent)
  821. {
  822. var pressed_key = null;
  823. if (theEvent.which) {
  824. pressed_key = theEvent.which;
  825. } else {
  826. pressed_key = theEvent.keyCode;
  827. }
  828. if (pressed_key == 13) { // enter pressed
  829. theForm.submit();
  830. return false;
  831. }
  832. return true;
  833. }',"\n\n";
  834. echo '// --></script>', "\n";
  835. }
  836. if ($this->display['form']) {
  837. echo '<form class="',$this->getCSSclass('form'),'" method="POST"';
  838. echo ' action="',$page_name,'" name="phpMyEdit_form">',"\n";
  839. }
  840. return true;
  841. } /* }}} */
  842. function form_end() /* {{{ */
  843. {
  844. if ($this->display['form']) {
  845. echo '</form>',"\n";
  846. }
  847. } /* }}} */
  848. function display_tab_labels($position) /* {{{ */
  849. {
  850. if (! is_array($this->tabs)) {
  851. return false;
  852. }
  853. echo '<table class="',$this->getCSSclass('tab', $position),'">',"\n";
  854. echo '<tr class="',$this->getCSSclass('tab', $position),'">',"\n";
  855. for ($i = 0; $i < count($this->tabs); $i++) {
  856. $css_class_name = $this->getCSSclass($i ? 'tab' : 'tab-selected', $position);
  857. echo '<td class="',$css_class_name,'" id="phpMyEdit_tab',$i,'_',$position,'_label">';
  858. echo '<a class="',$css_class_name,'" id="phpMyEdit_tab',$i,'_',$position,'_link';
  859. echo '" href="javascript:phpMyEdit_show_tab(\'phpMyEdit_tab',$i,'\')">';
  860. echo $this->tabs[$i],'</a></td>',"\n";
  861. }
  862. echo '<td class="',$this->getCSSclass('tab-end', $position),'">&nbsp;</td>',"\n";
  863. echo '</tr>',"\n";
  864. echo '</table>',"\n";
  865. } /* }}} */
  866. /*
  867. * Display functions
  868. */
  869. function display_add_record() /* {{{ */
  870. {
  871. for ($tab = 0, $k = 0; $k < $this->num_fds; $k++) {
  872. if (isset($this->fdd[$k]['tab']) && $this->tabs_enabled() && $k > 0) {
  873. $tab++;
  874. echo '</table>',"\n";
  875. echo '</div>',"\n";
  876. echo '<div id="phpMyEdit_tab',$tab,'">',"\n";
  877. echo '<table class="',$this->getCSSclass('main'),'" summary="',$this->tb,'">',"\n";
  878. }
  879. if (! $this->displayed[$k]) {
  880. continue;
  881. }
  882. if ($this->hidden($k)) {
  883. echo $this->htmlHidden($this->fds[$k], $this->fdd[$k]['default']);
  884. continue;
  885. }
  886. $css_postfix = @$this->fdd[$k]['css']['postfix'];
  887. $css_class_name = $this->getCSSclass('input', null, 'next', $css_postfix);
  888. echo '<tr class="',$this->getCSSclass('row', null, true, $css_postfix),'">',"\n";
  889. echo '<td class="',$this->getCSSclass('key', null, true, $css_postfix),'">',$this->fdd[$k]['name'],'</td>',"\n";
  890. echo '<td class="',$this->getCSSclass('value', null, true, $css_postfix),'"';
  891. echo $this->getColAttributes($k),">\n";
  892. if ($this->col_has_values($k)) {
  893. $vals = $this->set_values($k);
  894. $selected = @$this->fdd[$k]['default'];
  895. $multiple = $this->col_has_multiple_select($k);
  896. $readonly = $this->readonly($k);
  897. $strip_tags = true;
  898. $escape = true;
  899. echo $this->htmlSelect($this->fds[$k], $css_class_name, $vals, $selected,
  900. $multiple, $readonly, $strip_tags, $escape);
  901. } elseif (isset ($this->fdd[$k]['textarea'])) {
  902. echo '<textarea class="',$css_class_name,'" name="',$this->fds[$k],'"';
  903. echo ($this->readonly($k) ? ' disabled' : '');
  904. if (intval($this->fdd[$k]['textarea']['rows']) > 0) {
  905. echo ' rows="',$this->fdd[$k]['textarea']['rows'],'"';
  906. }
  907. if (intval($this->fdd[$k]['textarea']['cols']) > 0) {
  908. echo ' cols="',$this->fdd[$k]['textarea']['cols'],'"';
  909. }
  910. if (isset($this->fdd[$k]['textarea']['wrap'])) {
  911. echo ' wrap="',$this->fdd[$k]['textarea']['wrap'],'"';
  912. } else {
  913. echo ' wrap="virtual"';
  914. }
  915. echo '>',htmlspecialchars($this->fdd[$k]['default']),'</textarea>',"\n";
  916. } else {
  917. // Simple edit box required
  918. $size_ml_props = '';
  919. $maxlen = intval($this->fdd[$k]['maxlen']);
  920. $size = isset($this->fdd[$k]['size']) ? $this->fdd[$k]['size'] : min($maxlen, 60);
  921. $size && $size_ml_props .= ' size="'.$size.'"';
  922. $maxlen && $size_ml_props .= ' maxlength="'.$maxlen.'"';
  923. echo '<input class="',$css_class_name,'" type="text" ';
  924. echo ($this->readonly($k) ? 'disabled ' : ''),' name="',$this->fds[$k],'"';
  925. echo $size_ml_props,' value="';
  926. echo htmlspecialchars($this->fdd[$k]['default']),'">';
  927. }
  928. echo '</td>',"\n";
  929. if ($this->guidance) {
  930. $css_class_name = $this->getCSSclass('help', null, true, $css_postfix);
  931. $cell_value = $this->fdd[$k]['help'] ? $this->fdd[$k]['help'] : '&nbsp;';
  932. echo '<td class="',$css_class_name,'">',$cell_value,'</td>',"\n";
  933. }
  934. echo '</tr>',"\n";
  935. }
  936. } /* }}} */
  937. function display_copy_change_delete_record() /* {{{ */
  938. {
  939. /*
  940. * For delete or change: SQL SELECT to retrieve the selected record
  941. */
  942. $qparts['type'] = 'select';
  943. $qparts['select'] = $this->create_column_list();
  944. $qparts['from'] = $this->create_join_clause();
  945. $qparts['where'] = '('.$this->fqn($this->key).'='
  946. .$this->key_delim.$this->rec.$this->key_delim.')';
  947. $res = $this->myquery($this->query_make($qparts),__LINE__);
  948. if (! ($row = @mysql_fetch_array($res, MYSQL_ASSOC))) {
  949. return false;
  950. }
  951. for ($tab = 0, $k = 0; $k < $this->num_fds; $k++) {
  952. if (isset($this->fdd[$k]['tab']) && $this->tabs_enabled() && $k > 0) {
  953. $tab++;
  954. echo '</table>',"\n";
  955. echo '</div>',"\n";
  956. echo '<div id="phpMyEdit_tab',$tab,'">',"\n";
  957. echo '<table class="',$this->getCSSclass('main'),'" summary="',$this->tb,'">',"\n";
  958. }
  959. if (! $this->displayed[$k]) {
  960. continue;
  961. }
  962. if ($this->copy_operation() || $this->change_operation()) {
  963. if ($this->hidden($k)) {
  964. if ($k != $this->key_num) {
  965. echo $this->htmlHidden($this->fds[$k], $row["qf$k"]);
  966. }
  967. continue;
  968. }
  969. $css_postfix = @$this->fdd[$k]['css']['postfix'];
  970. echo '<tr class="',$this->getCSSclass('row', null, 'next', $css_postfix),'">',"\n";
  971. echo '<td class="',$this->getCSSclass('key', null, true, $css_postfix),'">',$this->fdd[$k]['name'],'</td>',"\n";
  972. /* There are two possibilities of readonly fields handling:
  973. 1. Display plain text for readonly timestamps and dates.
  974. 2. Display disabled input field
  975. In all cases particular readonly field will NOT be saved. */
  976. if ($this->col_has_datemask($k) && $this->readonly($k)) {
  977. echo $this->display_delete_field($row, $k);
  978. } elseif ($this->password($k)) {
  979. echo $this->display_password_field($row, $k);
  980. } else {
  981. echo $this->display_change_field($row, $k);
  982. }
  983. if ($this->guidance) {
  984. $css_class_name = $this->getCSSclass('help', null, true, $css_postfix);
  985. $cell_value = $this->fdd[$k]['help'] ? $this->fdd[$k]['help'] : '&nbsp;';
  986. echo '<td class="',$css_class_name,'">',$cell_value,'</td>',"\n";
  987. }
  988. echo '</tr>',"\n";
  989. } elseif ($this->delete_operation() || $this->view_operation()) {
  990. $css_postfix = @$this->fdd[$k]['css']['postfix'];
  991. echo '<tr class="',$this->getCSSclass('row', null, 'next', $css_postfix),'">',"\n";
  992. echo '<td class="',$this->getCSSclass('key', null, true, $css_postfix),'">',$this->fdd[$k]['name'],'</td>',"\n";
  993. if ($this->password($k)) {
  994. echo '<td class="',$this->getCSSclass('value', null, true, $css_postfix),'"';
  995. echo $this->getColAttributes($k),'>',$this->labels['hidden'],'</td>',"\n";
  996. } else {
  997. $this->display_delete_field($row, $k);
  998. }
  999. if ($this->guidance) {
  1000. $css_class_name = $this->getCSSclass('help', null, true, $css_postfix);
  1001. $cell_value = $this->fdd[$k]['help'] ? $this->fdd[$k]['help'] : '&nbsp;';
  1002. echo '<td class="',$css_class_name,'">',$cell_value,'</td>',"\n";
  1003. }
  1004. echo '</tr>',"\n";
  1005. }
  1006. }
  1007. } /* }}} */
  1008. function display_change_field($row, $k) /* {{{ */
  1009. {
  1010. $css_postfix = @$this->fdd[$k]['css']['postfix'];
  1011. $css_class_name = $this->getCSSclass('input', null, true, $css_postfix);
  1012. echo '<td class="',$this->getCSSclass('value', null, true, $css_postfix),'"';
  1013. echo $this->getColAttributes($k),">\n";
  1014. if ($this->col_has_values($k)) {
  1015. $vals = $this->set_values($k);
  1016. $multiple = $this->col_has_multiple_select($k);
  1017. $readonly = $this->readonly($k);
  1018. $strip_tags = true;
  1019. $escape = true;
  1020. echo $this->htmlSelect($this->fds[$k], $css_class_name, $vals, $row["qf$k"],
  1021. $multiple, $readonly, $strip_tags, $escape);
  1022. } elseif (isset($this->fdd[$k]['textarea'])) {
  1023. echo '<textarea class="',$css_class_name,'" name="',$this->fds[$k],'"';
  1024. echo ($this->readonly($k) ? ' disabled' : '');
  1025. if (intval($this->fdd[$k]['textarea']['rows']) > 0) {
  1026. echo ' rows="',$this->fdd[$k]['textarea']['rows'],'"';
  1027. }
  1028. if (intval($this->fdd[$k]['textarea']['cols']) > 0) {
  1029. echo ' cols="',$this->fdd[$k]['textarea']['cols'],'"';
  1030. }
  1031. if (isset($this->fdd[$k]['textarea']['wrap'])) {
  1032. echo ' wrap="',$this->fdd[$k]['textarea']['wrap'],'"';
  1033. } else {
  1034. echo ' wrap="virtual"';
  1035. }
  1036. echo '>',htmlspecialchars($row["qf$k"]),'</textarea>',"\n";
  1037. } else {
  1038. $size_ml_props = '';
  1039. $maxlen = intval($this->fdd[$k]['maxlen']);
  1040. $size = isset($this->fdd[$k]['size']) ? $this->fdd[$k]['size'] : min($maxlen, 60);
  1041. $size && $size_ml_props .= ' size="'.$size.'"';
  1042. $maxlen && $size_ml_props .= ' maxlength="'.$maxlen.'"';
  1043. echo '<input class="',$css_class_name,'" type="text" ';
  1044. echo ($this->readonly($k) ? 'disabled ' : ''),'name="',$this->fds[$k],'" value="';
  1045. echo htmlspecialchars($row["qf$k"]),'" ',$size_ml_props,'>',"\n";
  1046. }
  1047. echo '</td>',"\n";
  1048. } /* }}} */
  1049. function display_password_field($row, $k) /* {{{ */
  1050. {
  1051. $css_postfix = @$this->fdd[$k]['css']['postfix'];
  1052. echo '<td class="',$this->getCSSclass('value', null, true, $css_postfix),'"';
  1053. echo $this->getColAttributes($k),">\n";
  1054. $size_ml_props = '';
  1055. $maxlen = intval($this->fdd[$k]['maxlen']);
  1056. $size = isset($this->fdd[$k]['size']) ? $this->fdd[$k]['size'] : min($maxlen, 60);
  1057. $size && $size_ml_props .= ' size="'.$size.'"';
  1058. $maxlen && $size_ml_props .= ' maxlength="'.$maxlen.'"';
  1059. echo '<input class="',$this->getCSSclass('value', null, true, $css_postfix),'" type="password" ';
  1060. echo ($this->readonly($k) ? 'disabled ' : ''),'name="',$this->fds[$k],'" value="';
  1061. echo htmlspecialchars($row["qf$k"]),'" ',$size_ml_props,'>',"\n";
  1062. echo '</td>',"\n";
  1063. } /* }}} */
  1064. function display_delete_field($row, $k) /* {{{ */
  1065. {
  1066. $css_postfix = @$this->fdd[$k]['css']['postfix'];
  1067. $css_class_name = $this->getCSSclass('value', null, true, $css_postfix);
  1068. echo '<td class="',$css_class_name,'"',$this->getColAttributes($k),">\n";
  1069. echo $this->cellDisplay($k, $row, $css_class_name);
  1070. echo '</td>',"\n";
  1071. } /* }}} */
  1072. /**
  1073. * Returns CSS class name
  1074. */
  1075. function getCSSclass($name, $position = null, $divider = null, $postfix = null) /* {{{ */
  1076. {
  1077. static $div_idx = -1;
  1078. $elements = array($this->css['prefix'], $name);
  1079. if ($this->page_type && $this->css['page_type']) {
  1080. if ($this->page_type != 'L' && $this->page_type != 'F') {
  1081. $elements[] = $this->page_types[$this->page_type];
  1082. }
  1083. }
  1084. if ($position && $this->css['position']) {
  1085. $elements[] = $position;
  1086. }
  1087. if ($divider && $this->css['divider']) {
  1088. if ($divider === 'next') {
  1089. $div_idx++;
  1090. if ($this->css['divider'] > 0 && $div_idx >= $this->css['divider']) {
  1091. $div_idx = 0;
  1092. }
  1093. }
  1094. $elements[] = $div_idx;
  1095. }
  1096. if ($postfix) {
  1097. $elements[] = $postfix;
  1098. }
  1099. return join($this->css['separator'], $elements);
  1100. } /* }}} */
  1101. /**
  1102. * Returns field cell HTML attributes
  1103. */
  1104. function getColAttributes($k) /* {{{ */
  1105. {
  1106. $colattrs = '';
  1107. if (isset($this->fdd[$k]['colattrs'])) {
  1108. $colattrs .= ' ';
  1109. $colattrs .= trim($this->fdd[$k]['colattrs']);
  1110. }
  1111. if (isset($this->fdd[$k]['nowrap'])) {
  1112. $colattrs .= ' nowrap';
  1113. }
  1114. return $colattrs;
  1115. } /* }}} */
  1116. /**
  1117. * Substitutes variables in string
  1118. * (this is very simple but secure eval() replacement)
  1119. */
  1120. function substituteVars($str, $subst_ar) /* {{{ */
  1121. {
  1122. $array = preg_split('/\\$(\w+)/', $str, -1, PREG_SPLIT_DELIM_CAPTURE);
  1123. for ($i = 1; $i < count($array); $i += 2) {
  1124. if (isset($subst_ar[$array[$i]]))
  1125. $array[$i] = $subst_ar[$array[$i]];
  1126. }
  1127. return join('', $array);
  1128. } /* }}} */
  1129. /**
  1130. * Print URL
  1131. */
  1132. function urlDisplay($k, $link_val, $disp_val, $css, $key) /* {{{ */
  1133. {
  1134. $ret = '';
  1135. $name = $this->fds[$k];
  1136. $page = $this->page_name;
  1137. $url = 'rec='.$key.'&fm='.$this->fm.'&fl='.$this->fl;
  1138. $url .= '&qfn='.rawurlencode($this->qfn).$this->qfn;
  1139. $url .= '&'.$this->get_sfn_cgi_vars().$this->cgi['persist'];
  1140. $ar = array(
  1141. 'key' => $key,
  1142. 'name' => $name,
  1143. 'link' => $link_val,
  1144. 'value' => $disp_val,
  1145. 'css' => $css,
  1146. 'page' => $page,
  1147. 'url' => $url
  1148. );
  1149. $urllink = isset($this->fdd[$k]['URL'])
  1150. ? $this->substituteVars($this->fdd[$k]['URL'], $ar)
  1151. : $link_val;
  1152. $urldisp = isset($this->fdd[$k]['URLdisp'])
  1153. ? $this->substituteVars($this->fdd[$k]['URLdisp'], $ar)
  1154. : $disp_val;
  1155. $target = isset($this->fdd[$k]['URLtarget'])
  1156. ? 'target="'.htmlspecialchars($this->fdd[$k]['URLtarget']).'" '
  1157. : '';
  1158. $prefix_found = false;
  1159. $postfix_found = false;
  1160. $prefix_ar = @$this->fdd[$k]['URLprefix'];
  1161. $postfix_ar = @$this->fdd[$k]['URLpostfix'];
  1162. is_array($prefix_ar) || $prefix_ar = array($prefix_ar);
  1163. is_array($postfix_ar) || $postfix_ar = array($postfix_ar);
  1164. foreach ($prefix_ar as $prefix) {
  1165. if (! strncmp($prefix, $urllink, strlen($prefix))) {
  1166. $prefix_found = true;
  1167. break;
  1168. }
  1169. }
  1170. foreach ($postfix_ar as $postfix) {
  1171. if (! strncmp($postfix, $urllink, strlen($postfix))) {
  1172. $postfix_found = true;
  1173. break;
  1174. }
  1175. }
  1176. $prefix_found || $urllink = array_shift($prefix_ar).$urllink;
  1177. $postfix_found || $urllink = $urllink.array_shift($postfix_ar);
  1178. if (strlen($urllink) <= 0 || strlen($urldisp) <= 0) {
  1179. $ret = '&nbsp;';
  1180. } else {
  1181. $urllink = htmlspecialchars($urllink);
  1182. $urldisp = htmlspecialchars($urldisp);
  1183. $ret = '<a '.$target.'class="'.$css.'" href="'.$urllink.'">'.$urldisp.'</a>';
  1184. }
  1185. return $ret;
  1186. } /* }}} */
  1187. function cellDisplay($k, $row, $css) /* {{{ */
  1188. {
  1189. $escape = isset($this->fdd[$k]['escape']) ? $this->fdd[$k]['escape'] : true;
  1190. $key_rec = $row['qf'.$this->key_num];
  1191. if (@$this->fdd[$k]['datemask']) {
  1192. $value = intval($row["qf$k".'_timestamp']);
  1193. $value = $value ? @date($this->fdd[$k]['datemask'], $value) : '';
  1194. } else if (@$this->fdd[$k]['strftimemask']) {
  1195. $value = intval($row["qf$k".'_timestamp']);
  1196. $value = $value ? @strftime($this->fdd[$k]['strftimemask'], $value) : '';
  1197. } else if ($this->is_values2($k, $row["qf$k"])) {
  1198. $value = $row['qf'.$k.'_idx'];
  1199. if ($this->fdd[$k]['select'] == 'M') {
  1200. $value_ar = explode(',', $value);
  1201. $value_ar2 = array();
  1202. foreach ($value_ar as $value_key) {
  1203. if (isset($this->fdd[$k]['values2'][$value_key])) {
  1204. $value_ar2[$value_key] = $this->fdd[$k]['values2'][$value_key];
  1205. $escape = false;
  1206. }
  1207. }
  1208. $value = join(', ', $value_ar2);
  1209. } else {
  1210. if (isset($this->fdd[$k]['values2'][$value])) {
  1211. $value = $this->fdd[$k]['values2'][$value];
  1212. $escape = false;
  1213. }
  1214. }
  1215. } else {
  1216. $value = $row["qf$k"];
  1217. }
  1218. $original_value = $value;
  1219. if (@$this->fdd[$k]['strip_tags']) {
  1220. $value = strip_tags($value);
  1221. }
  1222. if (intval($this->fdd[$k]['trimlen']) > 0 && strlen($value) > $this->fdd[$k]['trimlen']) {
  1223. $value = ereg_replace("[\r\n\t ]+",' ',$value);
  1224. $value = substr($value, 0, $this->fdd[$k]['trimlen'] - 3).'...';
  1225. }
  1226. if (@$this->fdd[$k]['mask']) {
  1227. $value = sprintf($this->fdd[$k]['mask'], $value);
  1228. }
  1229. if (isset($this->fdd[$k]['eval'])) {
  1230. eval($this->fdd[$k]['eval']);
  1231. }
  1232. if ($this->col_has_URL($k)) {
  1233. return $this->urlDisplay($k, $original_value, $value, $css, $key_rec);
  1234. }
  1235. if (strlen($value) <= 0) {
  1236. return '&nbsp;';
  1237. }
  1238. if ($escape) {
  1239. $value = htmlspecialchars($value);
  1240. }
  1241. return nl2br($value);
  1242. } /* }}} */
  1243. /**
  1244. * Creates HTML submit input element
  1245. *
  1246. * @param name element name
  1247. * @param label key in the language hash used as label
  1248. * @param css_class_name CSS class name
  1249. * @param js_validation if add JavaScript validation subroutine to button
  1250. * @param disabled if mark the button as disabled
  1251. */
  1252. function htmlSubmit($name, $label, $css_class_name, $js_validation = true, $disabled = false) /* {{{ */
  1253. {
  1254. // Note that <input disabled> isn't valid HTML, but most browsers support it
  1255. $markdisabled = $disabled ? ' disabled' : '';
  1256. $ret = '<input'.$markdisabled.' type="submit" class="'.$css_class_name
  1257. .'" name="'.ltrim($markdisabled).$name
  1258. .'" value="'.(isset($this->labels[$label]) ? $this->labels[$label] : $label);
  1259. if ($js_validation) {
  1260. $ret .= '" onClick="return phpMyEdit_form_control(this.form);';
  1261. }
  1262. $ret .= '">';
  1263. return $ret;
  1264. } /* }}} */
  1265. /**
  1266. * Creates HTML hidden input element
  1267. *
  1268. * @param name element name
  1269. * @param value value
  1270. */
  1271. function htmlHidden($name, $value) /* {{{ */
  1272. {
  1273. return '<input type="hidden" name="'.htmlspecialchars($name)
  1274. .'" value="'.htmlspecialchars($value).'">'."\n";
  1275. } /* }}} */
  1276. /**
  1277. * Creates HTML select element (tag)
  1278. *
  1279. * @param name element name
  1280. * @param css CSS class name
  1281. * @param kv_array key => value array
  1282. * @param selected selected key (it can be single string, array of
  1283. * keys or multiple values separated by comma)
  1284. * @param multiple bool for multiple selection
  1285. * @param readonly bool for readonly/disabled selection
  1286. * @param strip_tags bool for stripping tags from values
  1287. * @param escape bool for HTML escaping values
  1288. */
  1289. function htmlSelect($name, $css, $kv_array, $selected = null, /* ...) {{{ */
  1290. /* booleans: */ $multiple = false, $readonly = false, $strip_tags = false, $escape = true)
  1291. {
  1292. $ret = '<select class="'.htmlspecialchars($css).'" name="'.htmlspecialchars($name);
  1293. if ($multiple) {
  1294. $ret .= '[]" multiple size="'.$this->multiple;
  1295. if (! is_array($selected) && $selected !== null) {
  1296. $selected = explode(',', $selected);
  1297. }
  1298. }
  1299. $ret .= '"'.($readonly ? ' disabled' : '').'>'."\n";
  1300. if (! is_array($selected)) {
  1301. $selected = $selected === null ? array() : array($selected);
  1302. }
  1303. $found = false;
  1304. foreach ($kv_array as $key => $value) {
  1305. $ret .= '<option value="'.htmlspecialchars($key).'"';
  1306. if ((! $found || $multiple) && in_array((string) $key, $selected, 1)
  1307. || (count($selected) == 0 && ! $found && ! $multiple)) {
  1308. $ret .= ' selected';
  1309. $found = true;
  1310. }
  1311. $strip_tags && $value = strip_tags($value);
  1312. $escape && $value = htmlspecialchars($value);
  1313. $ret .= '>'.$value.'</option>'."\n";
  1314. }
  1315. $ret .= '</select>';
  1316. return $ret;
  1317. } /* }}} */
  1318. /**
  1319. * Returns original variables HTML code for use in forms or links.
  1320. *
  1321. * @param mixed $origvars string or array of original varaibles
  1322. * @param string $method type of method ("POST" or "GET")
  1323. * @param mixed $default_value default value of variables
  1324. * if null, empty values will be skipped
  1325. * @return get HTML code of original varaibles
  1326. */
  1327. function get_origvars_html($origvars, $method = 'POST', $default_value = '') /* {{{ */
  1328. {
  1329. $ret = '';
  1330. $method = strtoupper($method);
  1331. if ($method == 'POST') {
  1332. if (! is_array($origvars)) {
  1333. $new_origvars = array();
  1334. foreach (explode('&', $origvars) as $param) {
  1335. $parts = explode('=', $param, 2);
  1336. if (! isset($parts[1])) {
  1337. $parts[1] = $default_value;
  1338. }
  1339. if (strlen($parts[0]) <= 0) {
  1340. continue;
  1341. }
  1342. $new_origvars[$parts[0]] = $parts[1];
  1343. }
  1344. $origvars =& $new_origvars;
  1345. }
  1346. foreach ($origvars as $key => $val) {
  1347. if (strlen($val) <= 0 && $default_value === null) {
  1348. continue;
  1349. }
  1350. $key = rawurldecode($key);
  1351. $val = rawurldecode($val);
  1352. $ret .= '<input type="hidden" name="';
  1353. $ret .= htmlspecialchars($key).'"';
  1354. $ret .= ' value="'.htmlspecialchars($val).'"';
  1355. $ret .= " />\n";
  1356. }
  1357. } else if (! strncmp('GET', $method, 3)) {
  1358. if (! is_array($origvars)) {
  1359. $ret .= $origvars;
  1360. } else {
  1361. foreach ($origvars as $key => $val) {
  1362. if (strlen($val) <= 0 && $default_value === null) {
  1363. continue;
  1364. }
  1365. $ret == '' || $ret .= '&amp;';
  1366. $ret .= htmlspecialchars(rawurlencode($key));
  1367. $ret .= '=';
  1368. $ret .= htmlspecialchars(rawurlencode($val));
  1369. }
  1370. }
  1371. if ($method[strlen($method) - 1] == '+') {
  1372. $ret = "?$ret";
  1373. }
  1374. } else {
  1375. trigger_error('Unsupported Platon::get_origvars_html() method: '
  1376. .$method, E_USER_ERROR);
  1377. }
  1378. return $ret;
  1379. } /* }}} */
  1380. function get_sfn_cgi_vars($alternative_sfn = null) /* {{{ */
  1381. {
  1382. if ($alternative_sfn === null) { // FAST! (cached return value)
  1383. static $ret = null;
  1384. $ret == null && $ret = $this->get_sfn_cgi_vars($this->sfn);
  1385. return $ret;
  1386. }
  1387. $ret = '';
  1388. $i = 0;
  1389. foreach ($alternative_sfn as $val) {
  1390. $ret != '' && $ret .= '&';
  1391. $ret .= "sfn[$i]=".rawurlencode($val);
  1392. $i++;
  1393. }
  1394. return $ret;
  1395. } /* }}} */
  1396. function get_cgi_var($name, $default_value = null) /* {{{ */
  1397. {
  1398. if (isset($this) && isset($this->cgi['overwrite'][$name])) {
  1399. return $this->cgi['overwrite'][$name];
  1400. }
  1401. static $magic_quotes_gpc = null;
  1402. if ($magic_quotes_gpc === null) {
  1403. $magic_quotes_gpc = get_magic_quotes_gpc();
  1404. }
  1405. global $HTTP_GET_VARS;
  1406. $var = @$HTTP_GET_VARS[$name];
  1407. if (! isset($var)) {
  1408. global $HTTP_POST_VARS;
  1409. $var = @$HTTP_POST_VARS[$name];
  1410. }
  1411. if (isset($var)) {
  1412. if ($magic_quotes_gpc) {
  1413. if (is_array($var)) {
  1414. foreach (array_keys($var) as $key) {
  1415. $var[$key] = stripslashes($var[$key]);
  1416. }
  1417. } else {
  1418. $var = stripslashes($var);
  1419. }
  1420. }
  1421. } else {
  1422. $var = @$default_value;
  1423. }
  1424. if (isset($this) && $var === null && isset($this->cgi['append'][$name])) {
  1425. return $this->cgi['append'][$name];
  1426. }
  1427. return $var;
  1428. } /* }}} */
  1429. function get_server_var($name) /* {{{ */
  1430. {
  1431. if ($name == 'REMOTE_USER') return $_SESSION['userid'];
  1432. if (isset($_SERVER[$name])) {
  1433. return $_SERVER[$name];
  1434. }
  1435. global $HTTP_SERVER_VARS;
  1436. if (isset($HTTP_SERVER_VARS[$name])) {
  1437. return $HTTP_SERVER_VARS[$name];
  1438. }
  1439. global $$name;
  1440. if (isset($$name)) {
  1441. return $$name;
  1442. }
  1443. return null;
  1444. } /* }}} */
  1445. /*
  1446. * Debug functions
  1447. */
  1448. function print_get_vars ($miss = 'No GET variables found') // debug only /* {{{ */
  1449. {
  1450. global $HTTP_GET_VARS;
  1451. // we parse form GET variables
  1452. if (is_array($HTTP_GET_VARS)) {
  1453. echo "<p> Variables per GET ";
  1454. foreach ($HTTP_GET_VARS as $k => $v) {
  1455. if (is_array($v)) {
  1456. foreach ($v as $akey => $aval) {
  1457. // $HTTP_GET_VARS[$k][$akey] = strip_tags($aval);
  1458. // $$k[$akey] = strip_tags($aval);
  1459. echo "$k\[$akey\]=$aval ";
  1460. }
  1461. } else {
  1462. // $HTTP_GET_VARS[$k] = strip_tags($val);
  1463. // $$k = strip_tags($val);
  1464. echo "$k=$v ";
  1465. }
  1466. }
  1467. echo '</p>';
  1468. } else {
  1469. echo '<p>';
  1470. echo $miss;
  1471. echo '</p>';
  1472. }
  1473. } /* }}} */
  1474. function print_post_vars($miss = 'No POST variables found') // debug only /* {{{ */
  1475. {
  1476. global $HTTP_POST_VARS;
  1477. // we parse form POST variables
  1478. if (is_array($HTTP_POST_VARS)) {
  1479. echo "<p>Variables per POST ";
  1480. foreach ($HTTP_POST_VARS as $k => $v) {
  1481. if (is_array($v)) {
  1482. foreach ($v as $akey => $aval) {
  1483. // $HTTP_POST_VARS[$k][$akey] = strip_tags($aval);
  1484. // $$k[$akey] = strip_tags($aval);
  1485. echo "$k\[$akey\]=$aval ";
  1486. }
  1487. } else {
  1488. // $HTTP_POST_VARS[$k] = strip_tags($val);
  1489. // $$k = strip_tags($val);
  1490. echo "$k=$v ";
  1491. }
  1492. }
  1493. echo '</p>';
  1494. } else {
  1495. echo '<p>';
  1496. echo $miss;
  1497. echo '</p>';
  1498. }
  1499. } /* }}} */
  1500. function print_vars ($miss = 'Current instance variables') // debug only /* {{{ */
  1501. {
  1502. echo "$miss ";
  1503. echo 'page_name=',$this->page_name,' ';
  1504. echo 'hn=',$this->hn,' ';
  1505. echo 'un=',$this->un,' ';
  1506. echo 'pw=',$this->pw,' ';
  1507. echo 'db=',$this->db,' ';
  1508. echo 'tb=',$this->tb,' ';
  1509. echo 'key=',$this->key,' ';
  1510. echo 'key_type=',$this->key_type,' ';
  1511. echo 'inc=',$this->inc,' ';
  1512. echo 'options=',$this->options,' ';
  1513. echo 'fdd=',$this->fdd,' ';
  1514. echo 'fl=',$this->fl,' ';
  1515. echo 'fm=',$this->fm,' ';
  1516. echo 'sfn=',htmlspecialchars($this->get_sfn_cgi_vars()),' ';
  1517. echo 'qfn=',$this->qfn,' ';
  1518. echo 'sw=',$this->sw,' ';
  1519. echo 'rec=',$this->rec,' ';
  1520. echo 'navop=',$this->navop,' ';
  1521. echo 'saveadd=',$this->saveadd,' ';
  1522. echo 'moreadd=',$this->moreadd,' ';
  1523. echo 'canceladd=',$this->canceladd,' ';
  1524. echo 'savechange=',$this->savechange,' ';
  1525. echo 'morechange=',$this->morechange,' ';
  1526. echo 'cancelchange=',$this->cancelchange,' ';
  1527. echo 'savedelete=',$this->savedelete,' ';
  1528. echo 'canceldelete=',$this->canceldelete,' ';
  1529. echo 'cancelview=',$this->cancelview,' ';
  1530. echo 'operation=',$this->operation,' ';
  1531. echo "\n";
  1532. } /* }}} */
  1533. /*
  1534. * Display buttons at top and bottom of page
  1535. */
  1536. function display_list_table_buttons($total_recs, $position) /* {{{ */
  1537. {
  1538. echo '<table class="',$this->getCSSclass('navigation', $position),'">',"\n";
  1539. echo '<tr class="',$this->getCSSclass('navigation', $position),'">',"\n";
  1540. echo '<td class="',$this->getCSSclass('buttons', $position),'">',"\n";
  1541. $listall = $this->inc <= 0; // Are we doing a listall?
  1542. $disabledprev = !($this->fm > 0 && !$listall);
  1543. $disablednext = !($this->fm + $this->inc < $total_recs && ! $listall);
  1544. $disabledgoto = !($listall || ($disablednext && $disabledprev)) ? '' : ' disabled';
  1545. $current_page = intval($this->fm / $this->inc) + 1;
  1546. $total_pages = max(1, ceil($total_recs / abs($this->inc)));
  1547. if (!$listall) {
  1548. echo $this->htmlSubmit('navop', 'First', $this->getCSSclass('first', $position), false, $disabledprev), '&nbsp;';
  1549. echo $this->htmlSubmit('navop', 'Prev', $this->getCSSclass('prev', $position), false, $disabledprev), '&nbsp;';
  1550. }
  1551. if ($this->add_enabled()) {
  1552. echo $this->htmlSubmit('operation', 'Add', $this->getCSSclass('add', $position), false, false), '&nbsp;';
  1553. }
  1554. if ($this->nav_buttons()) {
  1555. if ($this->view_enabled()) {
  1556. echo $this->htmlSubmit('operation', 'View', $this->getCSSclass('view', $position), false, !$total_recs), '&nbsp;';
  1557. }
  1558. if ($this->change_enabled()) {
  1559. echo $this->htmlSubmit('operation', 'Change', $this->getCSSclass('change', $position), false, !$total_recs), '&nbsp;';
  1560. }
  1561. if ($this->copy_enabled()) {
  1562. echo $this->htmlSubmit('operation', 'Copy', $this->getCSSclass('copy', $position), false, !$total_recs), '&nbsp;';
  1563. }
  1564. if ($this->delete_enabled()) {
  1565. echo $this->htmlSubmit('operation', 'Delete', $this->getCSSclass('delete', $position), false, !$total_recs), '&nbsp;';
  1566. }
  1567. }
  1568. if (!$listall) { //nav buttons are not displayed if explicit listall
  1569. echo $this->htmlSubmit('navop', 'Next', $this->getCSSclass('next', $position), false, $disablednext), '&nbsp;';
  1570. echo $this->htmlSubmit('navop', 'Last', $this->getCSSclass('last', $position), false, $disablednext), '&nbsp;';
  1571. echo $this->htmlSubmit('navop', 'Go to', $this->getCSSclass('goto', $position), false,
  1572. ($listall || ($disablednext && $disabledprev)));
  1573. echo '<select',$disabledgoto,' class="',$this->getCSSclass('goto', $position);
  1574. echo '" name="',ltrim($disabledgoto),'navfm',$position,'" onChange="return this.form.submit();">',"\n";
  1575. for ($i = 0; $i < $total_pages; $i++) {
  1576. echo '<option',($this->fm == $this->inc * $i) ? ' selected' : '';
  1577. echo ' value="',$this->inc * $i,'">',$i + 1,'</option>',"\n";
  1578. }
  1579. echo '</select>';
  1580. }
  1581. echo '</td>',"\n";
  1582. // Message is now written here
  1583. if (strlen(@$this->message) > 0) {
  1584. echo '<td class="',$this->getCSSclass('message', $position),'">',$this->message,'</td>',"\n";
  1585. }
  1586. // Display page and records statistics
  1587. echo '<td class="',$this->getCSSclass('stats', $position),'">',"\n";
  1588. if ($listall) {
  1589. echo $this->labels['Page'],':&nbsp;1&nbsp;',$this->labels['of'],'&nbsp;1';
  1590. } else {
  1591. echo $this->labels['Page'],':&nbsp;',$current_page;
  1592. echo '&nbsp;',$this->labels['of'],'&nbsp;',$total_pages;
  1593. }
  1594. echo '&nbsp; ',$this->labels['Records'],':&nbsp;',$total_recs;
  1595. echo '</td></tr></table>',"\n";
  1596. } /* }}} */
  1597. /*
  1598. * Display buttons at top and bottom of page
  1599. */
  1600. function display_record_buttons($position) /* {{{ */
  1601. {
  1602. echo '<table class="',$this->getCSSclass('navigation', $position),'">',"\n";
  1603. echo '<tr class="',$this->getCSSclass('navigation', $position),'">',"\n";
  1604. echo '<td class="',$this->getCSSclass('buttons', $position),'">',"\n";
  1605. if ($this->change_operation()) {
  1606. echo $this->htmlSubmit('savechange', 'Save', $this->getCSSclass('save', $position), true), '&nbsp;';
  1607. echo $this->htmlSubmit('morechange', 'Apply', $this->getCSSclass('more', $position), true), '&nbsp;';
  1608. echo $this->htmlSubmit('cancelchange', 'Cancel', $this->getCSSclass('cancel', $position), false);
  1609. } elseif ($this->add_operation()) {
  1610. echo $this->htmlSubmit('saveadd', 'Save', $this->getCSSclass('save', $position), true), '&nbsp;';
  1611. echo $this->htmlSubmit('moreadd', 'More', $this->getCSSclass('more', $position), true), '&nbsp;';
  1612. echo $this->htmlSubmit('canceladd', 'Cancel', $this->getCSSclass('cancel', $position), false);
  1613. } elseif ($this->copy_operation()) {
  1614. echo $this->htmlSubmit('saveadd', 'Save', $this->getCSSclass('save', $position), true), '&nbsp;';
  1615. echo $this->htmlSubmit('canceladd', 'Cancel', $this->getCSSclass('cancel', $position), false);
  1616. } elseif ($this->delete_operation()) {
  1617. echo $this->htmlSubmit('savedelete', 'Delete', $this->getCSSclass('save', $position), true), '&nbsp;';
  1618. echo $this->htmlSubmit('canceldelete', 'Cancel', $this->getCSSclass('cancel', $position), false);
  1619. } elseif ($this->view_operation()) {
  1620. if ($this->change_enabled()) {
  1621. echo $this->htmlSubmit('operation', 'Change', $this->getCSSclass('save', $position), true), '&nbsp;';
  1622. }
  1623. echo $this->htmlSubmit('cancelview', 'Cancel', $this->getCSSclass('cancel', $position), false);
  1624. }
  1625. // Message is now written here
  1626. echo '</td>',"\n";
  1627. if (strlen(@$this->message) > 0) {
  1628. echo '<td class="',$this->getCSSclass('message', $position),'">',$this->message,'</td>',"\n";
  1629. }
  1630. echo '</tr></table>',"\n";
  1631. } /* }}} */
  1632. /*
  1633. * Table Page Listing
  1634. */
  1635. function list_table() /* {{{ */
  1636. {
  1637. // Cancel Triggers
  1638. if ($this->add_canceled() && isset($this->triggers['insert']['cancel'])) {
  1639. include($this->triggers['insert']['cancel']);
  1640. }
  1641. if ($this->view_canceled() && isset($this->triggers['select']['cancel'])) {
  1642. include($this->triggers['select']['cancel']);
  1643. }
  1644. if ($this->change_canceled() && isset($this->triggers['update']['cancel'])) {
  1645. include($this->triggers['update']['cancel']);
  1646. }
  1647. if ($this->delete_canceled() && isset($this->triggers['delete']['cancel'])) {
  1648. include($this->triggers['delete']['cancel']);
  1649. }
  1650. if ($this->fm == '') {
  1651. $this->fm = 0;
  1652. }
  1653. if ($this->prev_operation()) {
  1654. $this->fm = $this->fm - $this->inc;
  1655. if ($this->fm < 0) {
  1656. $this->fm = 0;
  1657. }
  1658. }
  1659. if ($this->first_operation()) {
  1660. $this->fm = 0;
  1661. } // last operation must be performed below, after retrieving total_recs
  1662. if ($this->next_operation()) {
  1663. $this->fm += $this->inc;
  1664. }
  1665. if ($this->goto_operation()) {
  1666. $this->fm = $this->navfm;
  1667. }
  1668. // If sort sequence has changed, restart listing
  1669. $this->qfn != $this->prev_qfn && $this->fm = 0;
  1670. if (0) { // DEBUG
  1671. echo 'qfn vs. prev_qfn comparsion ';
  1672. echo '[<b>',htmlspecialchars($this->qfn),'</b>]';
  1673. echo '[<b>',htmlspecialchars($this->prev_qfn),'</b>]<br>';
  1674. echo 'comparsion <u>',($this->qfn == $this->prev_qfn ? 'proved' : 'failed'),'</u>';
  1675. echo '<hr>';
  1676. }
  1677. /*
  1678. * If user is allowed to Change/Delete records, we need an extra column
  1679. * to allow users to select a record
  1680. */
  1681. $select_recs = $this->key != '' &&
  1682. ($this->change_enabled() || $this->delete_enabled() || $this->view_enabled());
  1683. // Are we doing a listall?
  1684. $listall = $this->inc <= 0;
  1685. /*
  1686. * Display the MySQL table in an HTML table
  1687. */
  1688. $this->form_begin();
  1689. echo $this->get_origvars_html($this->get_sfn_cgi_vars());
  1690. echo '<input type="hidden" name="fl" value="',$this->fl,'">',"\n";
  1691. // Display buttons at top and/or bottom of page.
  1692. // Setup query to get num_rows. (sparky)
  1693. $total_recs = 0;
  1694. $count_parts = array(
  1695. 'type' => 'select',
  1696. 'select' => 'count(*)',
  1697. 'from' => $this->create_join_clause(),
  1698. 'where' => $this->make_where_from_query_opts());
  1699. $res = $this->myquery($this->query_make($count_parts), __LINE__);
  1700. $row = @mysql_fetch_array($res, MYSQL_NUM);
  1701. $total_recs = $row[0];
  1702. if ($this->last_operation()) {
  1703. $this->fm = (int)(($total_recs-1)/$this->inc)*$this->inc;
  1704. }
  1705. if ($this->nav_up()) {
  1706. $this->display_list_table_buttons($total_recs, 'up');
  1707. echo '<hr class="',$this->getCSSclass('hr', 'up'),'">',"\n";
  1708. }
  1709. if ($this->cgi['persist'] != '') {
  1710. echo $this->get_origvars_html($this->cgi['persist']);
  1711. }
  1712. if (! $this->filter_operation()) {
  1713. echo $this->get_origvars_html($this->qfn);
  1714. }
  1715. echo '<input type="hidden" name="qfn" value="',htmlspecialchars($this->qfn),'">',"\n";
  1716. echo '<input type="hidden" name="fm" value="',htmlspecialchars($this->fm),'">',"\n";
  1717. echo '<table class="',$this->getCSSclass('main'),'" summary="',$this->tb,'">',"\n";
  1718. echo '<tr class="',$this->getCSSclass('header'),'">',"\n";
  1719. /*
  1720. * System (navigation, selection) columns counting
  1721. */
  1722. $sys_cols = 0;
  1723. $sys_cols += intval($this->filter_enabled() || $select_recs);
  1724. if ($sys_cols > 0) {
  1725. $sys_cols += intval($this->nav_buttons()
  1726. && ($this->nav_text_links() || $this->nav_graphic_links()));
  1727. }
  1728. /*
  1729. * We need an initial column(s) (sys columns)
  1730. * if we have filters, Changes or Deletes enabled
  1731. */
  1732. if ($sys_cols) {
  1733. echo '<th class="',$this->getCSSclass('header'),'" colspan="',$sys_cols,'">';
  1734. if ($this->filter_enabled()) {
  1735. if ($this->filter_operation()) {
  1736. echo '<input class="',$this->getCSSclass('hide'),'" type="submit" name="sw" value="';
  1737. echo $this->labels['Hide'],'">';
  1738. echo '<input class="',$this->getCSSclass('clear'),'" type="submit" name="sw" value="';
  1739. echo $this->labels['Clear'],'">';
  1740. } else {
  1741. echo '<input class="',$this->getCSSclass('search'),'" type="submit" name="sw" value="';
  1742. echo $this->labels['Search'],'">';
  1743. }
  1744. } else {
  1745. echo '&nbsp;';
  1746. }
  1747. echo '</th>',"\n";
  1748. }
  1749. for ($k = 0; $k < $this->num_fds; $k++) {
  1750. $fd = $this->fds[$k];
  1751. if ($this->displayed[$k]) {
  1752. $css_postfix = @$this->fdd[$k]['css']['postfix'];
  1753. $css_class_name = $this->getCSSclass('header', null, null, $css_postfix);
  1754. $fdn = $this->fdd[$fd]['name'];
  1755. if (! $this->fdd[$fd]['sort'] || $this->password($fd)) {
  1756. echo '<th class="',$css_class_name,'">',$fdn,'</th>',"\n";
  1757. } else {
  1758. // Clicking on the current sort field reverses the sort order
  1759. $new_sfn = $this->sfn;
  1760. array_unshift($new_sfn, in_array("$k", $new_sfn, 1) ? "-$k" : $k);
  1761. echo '<th class="',$css_class_name,'">';
  1762. echo '<a class="',$css_class_name,'" href="';
  1763. echo htmlspecialchars($this->page_name.'?fm=0&fl='.$this->fl
  1764. .'&qfn='.rawurlencode($this->qfn).$this->qfn
  1765. .'&'.$this->get_sfn_cgi_vars($new_sfn).$this->cgi['persist']);
  1766. echo '">',$fdn,'</a></th>',"\n";
  1767. }
  1768. }
  1769. }
  1770. echo '</tr>',"\n";
  1771. /*
  1772. * Prepare the SQL Query from the data definition file
  1773. */
  1774. $qparts['type'] = 'select';
  1775. $qparts['select'] = $this->create_column_list();
  1776. // Even if the key field isn't displayed, we still need its value
  1777. if ($select_recs) {
  1778. if (!in_array ($this->key, $this->fds)) {
  1779. $qparts['select'] .= ','.$this->fqn($this->key);
  1780. }
  1781. }
  1782. $qparts['from'] = $this->create_join_clause();
  1783. $qparts['where'] = $this->make_where_from_query_opts();
  1784. // build up the ORDER BY clause
  1785. if (isset($this->sfn)) {
  1786. // WTF $raw_sort_fields?
  1787. //$raw_sort_fields = array();
  1788. $sort_fields = array();
  1789. $sort_fields_w = array();
  1790. foreach ($this->sfn as $field) {
  1791. if ($field[0] == '-') {
  1792. $field = substr($field, 1);
  1793. $desc = true;
  1794. } else {
  1795. $field = $field;
  1796. $desc = false;
  1797. }
  1798. //$raw_sort_field = 'qf'.$field;
  1799. $sort_field = $this->fqn($field);
  1800. $sort_field_w = $this->fdd[$field]['name'];
  1801. $this->col_has_sql($field) && $sort_field_w .= ' (sql)';
  1802. if ($desc) {
  1803. $sort_field .= ' DESC';
  1804. $sort_field_w .= ' '.$this->labels['descending'];
  1805. } else {
  1806. $sort_field_w .= ' '.$this->labels['ascending'];
  1807. }
  1808. //$raw_sort_fields[] = $raw_sort_field;
  1809. $sort_fields[] = $sort_field;
  1810. $sort_fields_w[] = $sort_field_w;
  1811. }
  1812. if (count($sort_fields) > 0) {
  1813. #$sort_fields = array_reverse($sort_fields);
  1814. #$sort_fields_w = array_reverse($sort_fields_w);
  1815. $qparts['orderby'] = join(',', $sort_fields);
  1816. }
  1817. }
  1818. $to = $this->fm + $this->inc;
  1819. if ($listall) {
  1820. #$qparts['limit'] = $this->fm.',-1';
  1821. } else {
  1822. $qparts['limit'] = $this->fm.','.$this->inc;
  1823. }
  1824. /*
  1825. * Main list_table() query
  1826. *
  1827. * Each row of the HTML table is one record from the SQL query. We must
  1828. * perform this query before filter printing, because we want to use
  1829. * mysql_field_len() function. We will also fetch the first row to get
  1830. * the field names.
  1831. */
  1832. $query = $this->query_make($qparts);
  1833. $res = $this->myquery($query, __LINE__);
  1834. if ($res == false) {
  1835. $this->error('invalid SQL query', $query);
  1836. return false;
  1837. }
  1838. $row = @mysql_fetch_array($res, MYSQL_ASSOC);
  1839. /* FILTER {{{
  1840. *
  1841. * Draw the filter and fill it with any data typed in last pass and stored
  1842. * in the array parameter keyword 'filter'. Prepare the SQL WHERE clause.
  1843. */
  1844. if ($this->filter_operation()) {
  1845. // Filter row retrieval
  1846. $fields = false;
  1847. $filter_row = $row;
  1848. if (! is_array($filter_row)) {
  1849. unset($qparts['where']);
  1850. $query = $this->query_make($qparts);
  1851. $res = $this->myquery($query, __LINE__);
  1852. if ($res == false) {
  1853. $this->error('invalid SQL query', $query);
  1854. return false;
  1855. }
  1856. $filter_row = @mysql_fetch_array($res, MYSQL_ASSOC);
  1857. }
  1858. /* Variable $fields is used to get index of particular field in
  1859. result. That index can be passed in example to mysql_field_len()
  1860. function. Use field names as indexes to $fields array. */
  1861. if (is_array($filter_row)) {
  1862. $fields = array_flip(array_keys($filter_row));
  1863. }
  1864. if ($fields != false) {
  1865. $css_class_name = $this->getCSSclass('filter');
  1866. echo '<tr class="',$css_class_name,'">',"\n";
  1867. echo '<td class="',$css_class_name,'" colspan="',$sys_cols,'">';
  1868. echo '<input class="',$this->getCSSclass('query'),'" type="submit" name="filter" value="';
  1869. echo $this->labels['Query'],'"></td>',"\n";
  1870. for ($k = 0; $k < $this->num_fds; $k++) {
  1871. if (! $this->displayed[$k]) {
  1872. continue;
  1873. }
  1874. $css_postfix = @$this->fdd[$k]['css']['postfix'];
  1875. $css_class_name = $this->getCSSclass('filter', null, null, $css_postfix);
  1876. $this->field_name = $this->fds[$k];
  1877. $fd = $this->field_name;
  1878. $this->field = $this->fdd[$fd];
  1879. $l = 'qf'.$k;
  1880. $lc = 'qf'.$k.'_comp';
  1881. $li = 'qf'.$k.'_id';
  1882. if ($this->clear_operation()) {
  1883. $m = null;
  1884. $mc = null;
  1885. $mi = null;
  1886. } else {
  1887. $m = $this->get_cgi_var($l);
  1888. $mc = $this->get_cgi_var($lc);
  1889. $mi = $this->get_cgi_var($li);
  1890. }
  1891. echo '<td class="',$css_class_name,'">';
  1892. if ($this->password($k)) {
  1893. echo '&nbsp;';
  1894. } else if ($this->fdd[$fd]['select'] == 'D' || $this->fdd[$fd]['select'] == 'M') {
  1895. // Multiple fields processing
  1896. // Default size is 2 and array required for values.
  1897. $from_table = ! $this->col_has_values($k) || isset($this->fdd[$k]['values']['table']);
  1898. $vals = $this->set_values($k, array('*' => '*'), null, $from_table);
  1899. $selected = $mi;
  1900. $multiple = $this->col_has_multiple_select($k);
  1901. $multiple |= $this->fdd[$fd]['select'] == 'M';
  1902. $readonly = false;
  1903. $strip_tags = true;
  1904. $escape = true;
  1905. echo $this->htmlSelect($l.'_id', $css_class_name, $vals,
  1906. $selected, $multiple, $readonly, $strip_tags, $escape);
  1907. } elseif ($this->fdd[$fd]['select'] == 'N' || $this->fdd[$fd]['select'] == 'T') {
  1908. $size_ml_props = '';
  1909. $maxlen = intval($this->fdd[$k]['maxlen']);
  1910. $maxlen > 0 || $maxlen = intval(@mysql_field_len($res, $fields["qf$k"]));
  1911. $size = isset($this->fdd[$k]['size']) ? $this->fdd[$k]['size']
  1912. : ($maxlen < 30 ? min($maxlen, 8) : 12);
  1913. $size && $size_ml_props .= ' size="'.$size.'"';
  1914. $maxlen && $size_ml_props .= ' maxlength="'.$maxlen.'"';
  1915. if ($this->fdd[$fd]['select'] == 'N') {
  1916. $mc = in_array($mc, $this->comp_ops) ? $mc : '=';
  1917. echo $this->htmlSelect($l.'_comp', $css_class_name, $this->comp_ops, $mc);
  1918. }
  1919. echo '<input class="',$css_class_name,'" value="';
  1920. echo htmlspecialchars(@$m),'" type="text" name="qf',$k,'"',$size_ml_props;
  1921. echo ' onKeyPress="return phpMyEdit_filter_handler(this.form, event);">';
  1922. } else {
  1923. echo '&nbsp;';
  1924. }
  1925. echo '</td>',"\n";
  1926. }
  1927. echo '</tr>',"\n";
  1928. }
  1929. } // }}}
  1930. /*
  1931. * Display sorting sequence
  1932. */
  1933. if ($qparts['orderby'] && $this->display['sort']) {
  1934. $css_class_name = $this->getCSSclass('sortinfo');
  1935. echo '<tr class="',$css_class_name,'">',"\n";
  1936. echo '<td class="',$css_class_name,'" colspan="',$sys_cols,'">';
  1937. echo '<a class="',$css_class_name,'" href="';
  1938. echo htmlspecialchars($this->get_server_var('PHP_SELF').'?fl='.$this->fl.'&fm='.$this->fm
  1939. .'&qfn='.rawurlencode($this->qfn).$this->qfn.$this->cgi['persist']);
  1940. echo '">',$this->labels['Clear'],'</a></td>',"\n";
  1941. echo '<td class="',$css_class_name,'" colspan="',$this->num_fields_displayed,'">';
  1942. echo $this->labels['Sorted By'],': ',join(', ', $sort_fields_w),'</td></tr>',"\n";
  1943. }
  1944. /*
  1945. * Display the current query
  1946. */
  1947. $text_query = $this->make_where_from_query_opts(null, true);
  1948. if ($text_query != '' && $this->display['query']) {
  1949. $css_class_name = $this->getCSSclass('queryinfo');
  1950. echo '<tr class="',$css_class_name,'">',"\n";
  1951. echo '<td class="',$css_class_name,'" colspan="',$sys_cols,'">';
  1952. echo '<a class="',$css_class_name,'" href="';
  1953. echo htmlspecialchars($this->get_server_var('PHP_SELF').'?fl='.$this->fl.'&fm='.$this->fm
  1954. .'&qfn='.rawurlencode($this->qfn).'&'.$this->get_sfn_cgi_vars().$this->cgi['persist']);
  1955. echo '">',$this->labels['Clear'],'</a></td>',"\n";
  1956. echo '<td class="',$css_class_name,'" colspan="',$this->num_fields_displayed,'">';
  1957. echo $this->labels['Current Query'],': ',htmlspecialchars($text_query),'</td></tr>',"\n";
  1958. }
  1959. if ($this->nav_text_links() || $this->nav_graphic_links()) {
  1960. $qstrparts = array();
  1961. strlen($this->fl) > 0 && $qstrparts[] = 'fl='.$this->fl;
  1962. strlen($this->fm) > 0 && $qstrparts[] = 'fm='.$this->fm;
  1963. count($this->sfn) > 0 && $qstrparts[] = $this->get_sfn_cgi_vars();
  1964. strlen($this->cgi['persist']) > 0 && $qstrparts[] = $this->cgi['persist'];
  1965. $qpview = $qstrparts;
  1966. $qpcopy = $qstrparts;
  1967. $qpchange = $qstrparts;
  1968. $qpdelete = $qstrparts;
  1969. $qpview[] = 'operation='.$this->labels['View'];
  1970. $qpcopy[] = 'operation='.$this->labels['Copy'];
  1971. $qpchange[] = 'operation='.urlencode($this->labels['Change']);
  1972. $qpdelete[] = 'operation='.urlencode($this->labels['Delete']);
  1973. $qpviewStr = '?'.join('&',$qpview).$this->qfn;
  1974. $qpcopyStr = '?'.join('&',$qpcopy).$this->qfn;
  1975. $qpchangeStr = '?'.join('&',$qpchange).$this->qfn;
  1976. $qpdeleteStr = '?'.join('&',$qpdelete).$this->qfn;
  1977. }
  1978. $fetched = true;
  1979. $first = true;
  1980. $rowCount = 0;
  1981. while ((!$fetched && ($row = @mysql_fetch_array($res, MYSQL_ASSOC)) != false)
  1982. || ($fetched && $row != false)) {
  1983. $fetched = false;
  1984. echo '<tr class="',$this->getCSSclass('row', null, 'next'),'">',"\n";
  1985. if ($sys_cols) { /* {{{ */
  1986. $key_rec = urlencode($row['qf'.$this->key_num]);
  1987. $qviewStr = $qpviewStr . '&rec='.$key_rec;
  1988. $qcopyStr = $qpcopyStr . '&rec='.$key_rec;
  1989. $qchangeStr = $qpchangeStr . '&rec='.$key_rec;
  1990. $qdeleteStr = $qpdeleteStr . '&rec='.$key_rec;
  1991. $css_class_name = $this->getCSSclass('navigation', null, true);
  1992. if ($select_recs) {
  1993. if (! $this->nav_buttons() || $sys_cols > 1) {
  1994. echo '<td class="',$css_class_name,'">';
  1995. }
  1996. if ($this->nav_graphic_links()) {
  1997. $printed_out = false;
  1998. if ($this->view_enabled()) {
  1999. $printed_out = true;
  2000. echo '<a class="',$css_class_name,'" href="';
  2001. echo htmlspecialchars($this->page_name.$qviewStr),'"><img class="';
  2002. echo $css_class_name,'" src="',$this->url['images'];
  2003. echo 'pme-view.png" height="15" width="16" border="0" alt="';
  2004. echo htmlspecialchars($this->labels['View']),'" title="';
  2005. echo htmlspecialchars($this->labels['View']),'"></a>';
  2006. }
  2007. if ($this->change_enabled()) {
  2008. $printed_out && print('&nbsp;');
  2009. $printed_out = true;
  2010. echo '<a class="',$css_class_name,'" href="';
  2011. echo htmlspecialchars($this->page_name.$qchangeStr),'"><img class="';
  2012. echo $css_class_name,'" src="',$this->url['images'];
  2013. echo 'pme-change.png" height="15" width="16" border="0" alt="';
  2014. echo htmlspecialchars($this->labels['Change']),'" title="';
  2015. echo htmlspecialchars($this->labels['Change']),'"></a>';
  2016. }
  2017. if ($this->copy_enabled()) {
  2018. $printed_out && print('&nbsp;');
  2019. $printed_out = true;
  2020. echo ' <a class="',$css_class_name,'" href="';
  2021. echo htmlspecialchars($this->page_name.$qcopyStr),'"><img class="';
  2022. echo $css_class_name,'" src="',$this->url['images'];
  2023. echo 'pme-copy.png" height="15" width="16" border="0" alt="';
  2024. echo htmlspecialchars($this->labels['Copy']),'" title="';
  2025. echo htmlspecialchars($this->labels['Copy']),'"></a>';
  2026. }
  2027. if ($this->delete_enabled($key_rec)) {
  2028. $printed_out && print('&nbsp;');
  2029. $printed_out = true;
  2030. echo '<a class="',$css_class_name,'" href="';
  2031. echo htmlspecialchars($this->page_name.$qdeleteStr),'"><img class="';
  2032. echo $css_class_name,'" src="',$this->url['images'];
  2033. echo 'pme-delete.png" height="15" width="16" border="0" alt="';
  2034. echo htmlspecialchars($this->labels['Delete']),'" title="';
  2035. echo htmlspecialchars($this->labels['Delete']),'"></a>';
  2036. }
  2037. }
  2038. if ($this->nav_text_links()) {
  2039. if ($this->nav_graphic_links()) {
  2040. echo '<br class="',$css_class_name,'">';
  2041. }
  2042. $printed_out = false;
  2043. if ($this->view_enabled()) {
  2044. $printed_out = true;
  2045. echo '<a class="',$css_class_name,'" href="';
  2046. echo htmlspecialchars($this->page_name.$qviewStr),'">V</a>&nbsp;';
  2047. }
  2048. if ($this->change_enabled()) {
  2049. $printed_out && print('&nbsp;');
  2050. $printed_out = true;
  2051. echo '<a class="',$css_class_name,'" href="';
  2052. echo htmlspecialchars($this->page_name.$qchangeStr),'">C</a>&nbsp;';
  2053. }
  2054. if ($this->copy_enabled()) {
  2055. $printed_out && print('&nbsp;');
  2056. $printed_out = true;
  2057. echo '<a class="',$css_class_name,'" href="';
  2058. echo htmlspecialchars($this->page_name.$qcopyStr),'">P</a>&nbsp;';
  2059. }
  2060. if ($this->delete_enabled($key_rec)) {
  2061. $printed_out && print('&nbsp;');
  2062. $printed_out = true;
  2063. echo '<a class="',$css_class_name,'" href="';
  2064. echo htmlspecialchars($this->page_name.$qdeleteStr),'">D</a>';
  2065. }
  2066. }
  2067. if (! $this->nav_buttons() || $sys_cols > 1) {
  2068. echo '</td>',"\n";
  2069. }
  2070. if ($this->nav_buttons()) {
  2071. echo '<td class="',$css_class_name,'"><input class="',$css_class_name;
  2072. echo '" type="radio" name="rec" value="',htmlspecialchars($key_rec),'"';
  2073. if ($first) {
  2074. echo ' checked';
  2075. $first = false;
  2076. }
  2077. echo '></td>',"\n";
  2078. }
  2079. } elseif ($this->filter_enabled()) {
  2080. echo '<td class="',$css_class_name,'" colspan=',$sys_cols,'>&nbsp;</td>',"\n";
  2081. }
  2082. } /* }}} */
  2083. for ($k = 0; $k < $this->num_fds; $k++) { /* {{{ */
  2084. $fd = $this->fds[$k];
  2085. if (! $this->displayed[$k]) {
  2086. continue;
  2087. }
  2088. $css_postfix = @$this->fdd[$k]['css']['postfix'];
  2089. $css_class_name = $this->getCSSclass('cell', null, true, $css_postfix);
  2090. if ($this->password($k)) {
  2091. echo '<td class="',$css_class_name,'">',$this->labels['hidden'],'</td>',"\n";
  2092. continue;
  2093. }
  2094. echo '<td class="',$css_class_name,'"',$this->getColAttributes($fd),'>';
  2095. echo $this->cellDisplay($k, $row, $css_class_name);
  2096. echo '</td>',"\n";
  2097. } /* }}} */
  2098. echo '</tr>',"\n";
  2099. }
  2100. /*
  2101. * Display and accumulate column aggregation info, do totalling query
  2102. * XXX this feature does not work yet!!!
  2103. */
  2104. // aggregates listing (if any)
  2105. if ($$var_to_total) {
  2106. // do the aggregate query if necessary
  2107. //if ($vars_to_total) {
  2108. $qp = array();
  2109. $qp['type'] = 'select';
  2110. $qp['select'] = $aggr_from_clause;
  2111. $qp['from'] = $this->create_join_clause ();
  2112. $qp['where'] = $this->make_where_from_query_opts();
  2113. $tot_query = $this->query_make($qp);
  2114. $totals_result = $this->myquery($tot_query,__LINE__);
  2115. $tot_row = @mysql_fetch_array($totals_result, MYSQL_ASSOC);
  2116. //}
  2117. $qp_aggr = $qp;
  2118. echo "\n",'<tr class="TODO-class">',"\n",'<td class="TODO-class">&nbsp;</td>',"\n";
  2119. /*
  2120. echo '<td>';
  2121. echo printarray($qp_aggr);
  2122. echo printarray($vars_to_total);
  2123. echo '</td>';
  2124. echo '<td colspan="'.($this->num_fds-1).'">'.$var_to_total.' '.$$var_to_total.'</td>';
  2125. */
  2126. // display the results
  2127. for ($k=0;$k<$this->num_fds;$k++) {
  2128. $fd = $this->fds[$k];
  2129. if (stristr($this->fdd[$fd]['options'],'L') or !isset($this->fdd[$fd]['options'])) {
  2130. echo '<td>';
  2131. $aggr_var = 'qf'.$k.'_aggr';
  2132. $$aggr_var = $this->get_cgi_var($aggr_var);
  2133. if ($$aggr_var) {
  2134. echo $this->sql_aggrs[$$aggr_var],': ',$tot_row[$aggr_var];
  2135. } else {
  2136. echo '&nbsp;';
  2137. }
  2138. echo '</td>',"\n";
  2139. }
  2140. }
  2141. echo '</tr>',"\n";
  2142. }
  2143. echo '</table>',"\n"; // end of table rows listing
  2144. if ($this->nav_down()) {
  2145. echo '<hr class="',$this->getCSSclass('hr', 'down'),'">',"\n";
  2146. $this->display_list_table_buttons($total_recs, 'down');
  2147. }
  2148. $this->form_end();
  2149. } /* }}} */
  2150. function display_record() /* {{{ */
  2151. {
  2152. // PRE Triggers
  2153. $ret = true;
  2154. if ($this->change_operation() && isset($this->triggers['update']['pre'])) {
  2155. $ret = include($this->triggers['update']['pre']);
  2156. // if PRE update fails, then back to view operation
  2157. if (! $ret) {
  2158. $this->operation = $this->labels['View'];
  2159. $ret = true;
  2160. }
  2161. }
  2162. if (($this->add_operation() || $this->copy_operation())
  2163. && isset($this->triggers['insert']['pre'])) {
  2164. $ret = include($this->triggers['insert']['pre']);
  2165. }
  2166. if ($this->view_operation() && isset($this->triggers['select']['pre'])) {
  2167. $ret = include($this->triggers['select']['pre']);
  2168. }
  2169. if ($this->delete_operation() && isset($this->triggers['delete']['pre'])) {
  2170. $ret = include($this->triggers['delete']['pre']);
  2171. }
  2172. // if PRE insert/view/delete fail, then back to the list
  2173. if ($ret == false) {
  2174. $this->operation = '';
  2175. $this->list_table();
  2176. return;
  2177. }
  2178. $this->form_begin();
  2179. if ($this->cgi['persist'] != '') {
  2180. echo $this->get_origvars_html($this->cgi['persist']);
  2181. }
  2182. echo $this->get_origvars_html($this->get_sfn_cgi_vars());
  2183. echo $this->get_origvars_html($this->qfn);
  2184. echo '<input type="hidden" name="qfn" value="',htmlspecialchars($this->qfn),'">',"\n";
  2185. echo '<input type="hidden" name="rec" value="',($this->copy_operation()?'':$this->rec),'">',"\n";
  2186. echo '<input type="hidden" name="fm" value="',$this->fm,'">',"\n";
  2187. echo '<input type="hidden" name="fl" value="',$this->fl,'">',"\n";
  2188. if ($this->nav_up()) {
  2189. $this->display_record_buttons('up');
  2190. echo '<hr class="',$this->getCSSclass('hr', 'up'),'">',"\n";
  2191. if ($this->tabs_enabled()) {
  2192. $this->display_tab_labels('up');
  2193. }
  2194. }
  2195. if ($this->tabs_enabled()) {
  2196. echo '<div id="phpMyEdit_tab0">',"\n";
  2197. }
  2198. echo '<table class="',$this->getCSSclass('main'),'" summary="',$this->tb,'">',"\n";
  2199. if ($this->add_operation()) {
  2200. $this->display_add_record();
  2201. } else {
  2202. $this->display_copy_change_delete_record();
  2203. }
  2204. echo '</table>',"\n";
  2205. echo '</div>',"\n";
  2206. if ($this->nav_down()) {
  2207. if ($this->tabs_enabled()) {
  2208. $this->display_tab_labels('down');
  2209. }
  2210. echo '<hr class="',$this->getCSSclass('hr', 'down'),'">',"\n";
  2211. $this->display_record_buttons('down');
  2212. }
  2213. $this->form_end();
  2214. } /* }}} */
  2215. /*
  2216. * Action functions
  2217. */
  2218. function do_add_record() /* {{{ */
  2219. {
  2220. // Preparing query
  2221. $query = '';
  2222. $key_col_val = '';
  2223. for ($k = 0; $k < $this->num_fds; $k++) {
  2224. if ($this->processed($k)) {
  2225. }
  2226. }
  2227. $vals_quoted = array();
  2228. $newvals = array();
  2229. for ($k = 0; $k < $this->num_fds; $k++) {
  2230. if ($this->processed($k)) {
  2231. $fd = $this->fds[$k];
  2232. if ($this->readonly($k)) {
  2233. $fn = (string) @$this->fdd[$k]['default'];
  2234. } else {
  2235. $fn = $this->get_cgi_var($this->fds[$k]);
  2236. }
  2237. if ($fd == $this->key) {
  2238. $key_col_val = $fn;
  2239. }
  2240. $xfd = $fd;
  2241. if (in_array($fd, array('sql'))) $xfd = "`".$fd."`";
  2242. if ($query == '') {
  2243. $query = 'INSERT INTO '.$this->tb.' ('.$xfd; // )
  2244. } else {
  2245. $query .= ','.$xfd;
  2246. }
  2247. $newvals[$this->fds[$k]] = is_array($fn) ? join(',',$fn) : $fn;
  2248. if ($this->col_has_sqlw($k)) {
  2249. $val = $newvals[$this->fds[$k]];
  2250. $val_as = addslashes($val);
  2251. $val_qas = '"'.addslashes($val).'"';
  2252. $vals_quoted[$k] = $this->substituteVars(
  2253. $this->fdd[$k]['sqlw'], array(
  2254. 'val_qas' => $val_qas,
  2255. 'val_as' => $val_as,
  2256. 'val' => $val
  2257. ));
  2258. } else {
  2259. $vals_quoted[$k] = addslashes($newvals[$this->fds[$k]]);
  2260. $vals_quoted[$k] = "'".$vals_quoted[$k]."'";
  2261. }
  2262. }
  2263. }
  2264. // Creating array of changed keys ($changed)
  2265. $changed = array_keys($newvals);
  2266. // Before trigger
  2267. if (isset($this->triggers['insert']['before'])) {
  2268. $ret = include($this->triggers['insert']['before']);
  2269. if ($ret == false) {
  2270. return false;
  2271. }
  2272. }
  2273. // Real query (no additional query in this method)
  2274. $query .= ') VALUES ('.join(',',$vals_quoted).')'; // )
  2275. $res = $this->myquery($query, __LINE__);
  2276. $this->message = @mysql_affected_rows($this->dbh).' '.$this->labels['record added'];
  2277. if (! $res) {
  2278. return false;
  2279. }
  2280. $this->insert_id = mysql_insert_id($this->dbh);
  2281. // Notify list
  2282. if (@$this->notify['insert'] || @$this->notify['all']) {
  2283. $this->email_notify(false, $newvals);
  2284. }
  2285. // Note change in log table
  2286. if ($this->logtable) {
  2287. $query = sprintf('INSERT INTO %s'
  2288. .' (updated, user, host, operation, tab, rowkey, col, oldval, newval)'
  2289. .' VALUES (NOW(), "%s", "%s", "insert", "%s", "%s", "", "", "%s")',
  2290. $this->logtable, addslashes($this->get_server_var('REMOTE_USER')),
  2291. addslashes($this->get_server_var('REMOTE_ADDR')), addslashes($this->tb),
  2292. addslashes($key_col_val), addslashes(serialize($newvals)));
  2293. $this->myquery($query, __LINE__);
  2294. }
  2295. // After trigger
  2296. if (isset($this->triggers['insert']['after'])) {
  2297. $ret = include($this->triggers['insert']['after']);
  2298. if ($ret == false) {
  2299. return false;
  2300. }
  2301. }
  2302. return true;
  2303. } /* }}} */
  2304. function do_change_record() /* {{{ */
  2305. {
  2306. // Preparing queries
  2307. $query_real = '';
  2308. $query_oldrec = '';
  2309. $newvals = array();
  2310. $oldvals = array();
  2311. $changed = array();
  2312. for ($k = 0; $k < $this->num_fds; $k++) {
  2313. if ($this->processed($k) && !$this->readonly($k)) {
  2314. $fd = $this->fds[$k];
  2315. $fn = $this->get_cgi_var($fd);
  2316. $newvals[$this->fds[$k]] = is_array($fn) ? join(',',$fn) : $fn;
  2317. if ($this->col_has_sqlw($k)) {
  2318. $val = $newvals[$this->fds[$k]];
  2319. $val_as = addslashes($val);
  2320. $val_qas = '"'.addslashes($val).'"';
  2321. $newValue = $this->substituteVars(
  2322. $this->fdd[$k]['sqlw'], array(
  2323. 'val_qas' => $val_qas,
  2324. 'val_as' => $val_as,
  2325. 'val' => $val
  2326. ));
  2327. } else {
  2328. $newValue = $newvals[$this->fds[$k]];
  2329. $newValue = "'".addslashes($newValue)."'";
  2330. }
  2331. $xfd = $fd;
  2332. if (in_array($fd, array('sql'))) $xfd = "`".$fd."`";
  2333. if ($query_real == '') {
  2334. $query_real = 'UPDATE '.$this->tb.' SET '.$xfd.'='.$newValue;
  2335. $query_oldrec = 'SELECT '.$xfd;
  2336. } else {
  2337. $query_real .= ','.$xfd.'='.$newValue;
  2338. $query_oldrec .= ','.$xfd;
  2339. }
  2340. }
  2341. }
  2342. $where_part = " WHERE (".$this->key.'='.$this->key_delim.$this->rec.$this->key_delim.')';
  2343. if ($query_real) $query_real .= $where_part;
  2344. if ($query_oldrec == '') $query_oldrec = 'SELECT * ';
  2345. $query_oldrec .= ' FROM ' . $this->tb . $where_part;
  2346. // Additional query (must go before real query)
  2347. $res = $this->myquery($query_oldrec, __LINE__);
  2348. $oldvals = @mysql_fetch_array($res, MYSQL_ASSOC);
  2349. @mysql_free_result($res);
  2350. // Creating array of changed keys ($changed)
  2351. for ($k = 0; $k < $this->num_fds; $k++) {
  2352. $key = $this->fds[$k];
  2353. if (! $this->processed($k) || $this->readonly($k) || $oldvals[$key] == $newvals[$key]) {
  2354. continue;
  2355. }
  2356. $changed[] = $key;
  2357. }
  2358. // Before trigger
  2359. if (isset($this->triggers['update']['before'])) {
  2360. $ret = include($this->triggers['update']['before']);
  2361. if ($ret == false) {
  2362. return false;
  2363. }
  2364. }
  2365. // Real query
  2366. if ($query_real) {
  2367. $res = $this->myquery($query_real, __LINE__);
  2368. $this->message = @mysql_affected_rows($this->dbh).' '.$this->labels['record changed'];
  2369. if (! $res) {
  2370. return false;
  2371. }
  2372. }
  2373. // Another additional query (must go after real query)
  2374. $res = $this->myquery($query_oldrec, __LINE__);
  2375. $newvals = @mysql_fetch_array($res, MYSQL_ASSOC);
  2376. @mysql_free_result($res);
  2377. // Creating array of changed keys ($changed)
  2378. $changed = array();
  2379. for ($k = 0; $k < $this->num_fds; $k++) {
  2380. $key = $this->fds[$k];
  2381. if (! $this->processed($k) || $this->readonly($k) || $oldvals[$key] == $newvals[$key]) {
  2382. continue;
  2383. }
  2384. $changed[] = $key;
  2385. }
  2386. // Notify list
  2387. if (@$this->notify['update'] || @$this->notify['all']) {
  2388. if (count($changed) > 0) {
  2389. $this->email_notify($oldvals, $newvals);
  2390. }
  2391. }
  2392. // Note change in log table
  2393. if ($this->logtable) {
  2394. foreach ($changed as $key) {
  2395. $qry = sprintf('INSERT INTO %s'
  2396. .' (updated, user, host, operation, tab, rowkey, col, oldval, newval)'
  2397. .' VALUES (NOW(), "%s", "%s", "update", "%s", "%s", "%s", "%s", "%s")',
  2398. $this->logtable, addslashes($this->get_server_var('REMOTE_USER')),
  2399. addslashes($this->get_server_var('REMOTE_ADDR')), addslashes($this->tb),
  2400. addslashes($this->rec), addslashes($key),
  2401. addslashes($oldvals[$key]), addslashes($newvals[$key]));
  2402. $this->myquery($qry, __LINE__);
  2403. }
  2404. }
  2405. // After trigger
  2406. if (isset($this->triggers['update']['after'])) {
  2407. $ret = include($this->triggers['update']['after']);
  2408. if ($ret == false) {
  2409. return false;
  2410. }
  2411. }
  2412. return true;
  2413. } /* }}} */
  2414. function do_delete_record() /* {{{ */
  2415. {
  2416. // Additional query
  2417. $query = 'SELECT * FROM '.$this->tb.' WHERE ('.$this->key.' = '
  2418. .$this->key_delim.$this->rec.$this->key_delim.')'; // )
  2419. $res = $this->myquery($query, __LINE__);
  2420. $oldvals = @mysql_fetch_array($res, MYSQL_ASSOC);
  2421. @mysql_free_result($res);
  2422. // Creating array of changed keys ($changed)
  2423. if (!is_array($oldvals)) $oldvals = array();
  2424. $changed = array_keys($oldvals);
  2425. // Before trigger
  2426. if (isset($this->triggers['delete']['before'])) {
  2427. $ret = include($this->triggers['delete']['before']);
  2428. if ($ret == false) {
  2429. return false;
  2430. }
  2431. }
  2432. // Real query
  2433. $query = 'DELETE FROM '.$this->tb.' WHERE ('.$this->key.' = '
  2434. .$this->key_delim.$this->rec.$this->key_delim.')'; // )
  2435. $res = $this->myquery($query, __LINE__);
  2436. $this->message = @mysql_affected_rows($this->dbh).' '.$this->labels['record deleted'];
  2437. if (! $res) {
  2438. return false;
  2439. }
  2440. // Notify list
  2441. if (@$this->notify['delete'] || @$this->notify['all']) {
  2442. $this->email_notify($oldvals, false);
  2443. }
  2444. // Note change in log table
  2445. if ($this->logtable) {
  2446. $query = sprintf('INSERT INTO %s'
  2447. .' (updated, user, host, operation, tab, rowkey, col, oldval, newval)'
  2448. .' VALUES (NOW(), "%s", "%s", "delete", "%s", "%s", "%s", "%s", "")',
  2449. $this->logtable, addslashes($this->get_server_var('REMOTE_USER')),
  2450. addslashes($this->get_server_var('REMOTE_ADDR')), addslashes($this->tb),
  2451. addslashes($this->rec), addslashes($key), addslashes(serialize($oldvals)));
  2452. $this->myquery($query, __LINE__);
  2453. }
  2454. // After trigger
  2455. if (isset($this->triggers['delete']['after'])) {
  2456. $ret = include($this->triggers['delete']['after']);
  2457. if ($ret == false) {
  2458. return false;
  2459. }
  2460. }
  2461. return true;
  2462. } /* }}} */
  2463. function email_notify($old_vals, $new_vals) /* {{{ */
  2464. {
  2465. if (! function_exists('mail')) {
  2466. return false;
  2467. }
  2468. if ($old_vals != false && $new_vals != false) {
  2469. $action = 'update';
  2470. $subject = 'Record updated in';
  2471. $body = 'An item with '.$this->fdd[$this->key]['name'].' = '
  2472. .$this->key_delim.$this->rec.$this->key_delim .' was updated in';
  2473. $vals = $new_vals;
  2474. } elseif ($new_vals != false) {
  2475. $action = 'insert';
  2476. $subject = 'Record added to';
  2477. $body = 'A new item was added into';
  2478. $vals = $new_vals;
  2479. } elseif ($old_vals != false) {
  2480. $action = 'delete';
  2481. $subject = 'Record deleted from';
  2482. $body = 'An item was deleted from';
  2483. $vals = $old_vals;
  2484. } else {
  2485. return false;
  2486. }
  2487. $addr = $this->get_server_var('REMOTE_ADDR');
  2488. $user = $this->get_server_var('REMOTE_USER');
  2489. $body = 'This notification e-mail was automatically generated by phpMyEdit.'."\n\n".$body;
  2490. $body .= ' table '.$this->tb.' in MySQL database '.$this->db.' on '.$this->page_name;
  2491. $body .= ' by '.($user == '' ? 'unknown user' : "user $user").' from '.$addr;
  2492. $body .= ' at '.date('d/M/Y H:i').' with the following fields:'."\n\n";
  2493. $i = 1;
  2494. foreach ($vals as $k => $text) {
  2495. $name = isset($this->fdd[$k]['name~'])
  2496. ? $this->fdd[$k]['name~'] : $this->fdd[$k]['name'];
  2497. if ($action == 'update') {
  2498. if ($old_vals[$k] == $new_vals[$k]) {
  2499. continue;
  2500. }
  2501. $body .= sprintf("[%02s] %s (%s)\n WAS: %s\n IS: %s\n",
  2502. $i, $name, $k, $old_vals[$k], $new_vals[$k]);
  2503. } else {
  2504. $body .= sprintf('[%02s] %s (%s): %s'."\n", $i, $name, $k, $text);
  2505. }
  2506. $i++;
  2507. }
  2508. $body .= "\n--\r\n"; // \r is needed for signature separating
  2509. $body .= "phpMyEdit\ninstant MySQL table editor and code generator\n";
  2510. $body .= "http://www.platon.sk/projects/phpMyEdit/\n\n";
  2511. $subject = @$this->notify['prefix'].$subject.' '.$this->db.'.'.$this->tb;
  2512. $subject = trim($subject); // just for sure
  2513. $wrap_w = intval(@$this->notify['wrap']);
  2514. $wrap_w > 0 || $wrap_w = 72;
  2515. $from = (string) @$this->notify['from'];
  2516. $from != '' || $from = 'webmaster@'.strtolower($this->get_server_var('SERVER_NAME'));
  2517. $headers = 'From: '.$from."\n".'X-Mailer: PHP/'.phpversion().' (phpMyEdit)';
  2518. $body = wordwrap($body, $wrap_w, "\n", 1);
  2519. $emails = (array) $this->notify[$action] + (array) $this->notify['all'];
  2520. foreach ($emails as $email) {
  2521. if (! empty($email)) {
  2522. mail(trim($email), $subject, $body, $headers);
  2523. }
  2524. }
  2525. return true;
  2526. } /* }}} */
  2527. /*
  2528. * Recreate functions
  2529. */
  2530. function recreate_fdd() /* {{{ */
  2531. {
  2532. // TODO: one level deeper browsing
  2533. $this->page_type = 'L'; // list by default
  2534. $this->filter_operation() && $this->page_type = 'F';
  2535. $this->view_operation() && $this->page_type = 'V';
  2536. $this->delete_operation() && $this->page_type = 'D';
  2537. $this->add_operation() && $this->page_type = 'A';
  2538. $this->change_operation() && $this->page_type = 'C';
  2539. $this->copy_operation() && $this->page_type = 'P';
  2540. // Restore backups (if exists)
  2541. foreach (array_keys($this->fdd) as $column) {
  2542. foreach (array_keys($this->fdd[$column]) as $col_option) {
  2543. if ($col_option[strlen($col_option) - 1] != '~')
  2544. continue;
  2545. $this->fdd[$column][substr($col_option, 0, strlen($col_option) - 1)]
  2546. = $this->fdd[$column][$col_option];
  2547. unset($this->fdd[$column][$col_option]);
  2548. }
  2549. }
  2550. foreach (array_keys($this->fdd) as $column) {
  2551. foreach (array_keys($this->fdd[$column]) as $col_option) {
  2552. if (! strchr($col_option, '|')) {
  2553. continue;
  2554. }
  2555. $col_ar = explode('|', $col_option, 2);
  2556. if (! stristr($col_ar[1], $this->page_type)) {
  2557. continue;
  2558. }
  2559. // Make field backups
  2560. $this->fdd[$column][$col_ar[0] .'~'] = $this->fdd[$column][$col_ar[0]];
  2561. $this->fdd[$column][$col_option.'~'] = $this->fdd[$column][$col_option];
  2562. // Set particular field
  2563. $this->fdd[$column][$col_ar[0]] = $this->fdd[$column][$col_option];
  2564. unset($this->fdd[$column][$col_option]);
  2565. }
  2566. }
  2567. } /* }}} */
  2568. function recreate_displayed() /* {{{ */
  2569. {
  2570. $field_num = 0;
  2571. $num_fields_displayed = 0;
  2572. $this->fds = array();
  2573. $this->displayed = array();
  2574. $this->guidance = false;
  2575. foreach (array_keys($this->fdd) as $key) {
  2576. if (preg_match('/^\d*$/', $key)) { // skipping numeric keys
  2577. continue;
  2578. }
  2579. $this->fds[$field_num] = $key;
  2580. /* We must use here displayed() function, because displayed[] array
  2581. is not created yet. We will simultaneously create that array as well. */
  2582. if ($this->displayed[$field_num] = $this->displayed($field_num)) {
  2583. $num_fields_displayed++;
  2584. }
  2585. if (is_array(@$this->fdd[$key]['values']) && ! isset($this->fdd[$key]['values']['table'])) {
  2586. $this->fdd[$key]['values2'] = array();
  2587. foreach ($this->fdd[$key]['values'] as $val) {
  2588. $this->fdd[$key]['values2'][$val] = $val;
  2589. }
  2590. unset($this->fdd[$key]['values']);
  2591. }
  2592. isset($this->fdd[$key]['help']) && $this->guidance = true;
  2593. $this->fdd[$field_num] = $this->fdd[$key];
  2594. $field_num++;
  2595. }
  2596. $this->num_fds = $field_num;
  2597. $this->num_fields_displayed = $num_fields_displayed;
  2598. $this->key_num = array_search($this->key, $this->fds);
  2599. /* Adds first displayed column into sorting fields by replacing last
  2600. array entry. Also remove duplicite values and change column names to
  2601. their particular field numbers.
  2602. Note that entries like [0]=>'9' [1]=>'-9' are correct and they will
  2603. have desirable sorting behaviour. So there is no need to remove them.
  2604. */
  2605. for ($k = 0; ! $this->displayed[$k]; $k++);
  2606. #if (count($this->sfn)>0) $this->sfn[count($this->sfn) - 1] = "$k"; // important quotes
  2607. if (!is_array($this->sfn)) $this->sfn = array();
  2608. $this->sfn = array_unique($this->sfn);
  2609. $check_ar = array();
  2610. foreach ($this->sfn as $key => $val) {
  2611. if (preg_match('/^[-]?\d*$/', $val)) { // skipping numeric keys
  2612. $val = abs($val);
  2613. if (in_array($val, $check_ar) || $this->password($val)) {
  2614. unset($this->sfn[$key]);
  2615. } else {
  2616. $check_ar[] = $val;
  2617. }
  2618. continue;
  2619. }
  2620. if ($val[0] == '-') {
  2621. $val = substr($val, 1);
  2622. $minus = '-';
  2623. } else {
  2624. $minus = '';
  2625. }
  2626. if (($val = array_search($val, $this->fds)) === false || $this->password($val)) {
  2627. unset($this->sfn[$key]);
  2628. } else {
  2629. $val = intval($val);
  2630. if (in_array($val, $check_ar)) {
  2631. unset($this->sfn[$key]);
  2632. } else {
  2633. $this->sfn[$key] = $minus.$val;
  2634. $check_ar[] = $val;
  2635. }
  2636. }
  2637. }
  2638. $this->sfn = array_unique($this->sfn);
  2639. return true;
  2640. } /* }}} */
  2641. /*
  2642. * Error handling function
  2643. */
  2644. function error($message, $additional_info = '') /* {{{ */
  2645. {
  2646. echo '<h1>phpMyEdit error: ',htmlspecialchars($message),'</h1>',"\n";
  2647. if ($additional_info != '') {
  2648. echo '<hr>',htmlspecialchars($additional_info);
  2649. }
  2650. return false;
  2651. } /* }}} */
  2652. /*
  2653. * Database connection function
  2654. */
  2655. function connect() /* {{{ */
  2656. {
  2657. if (!isset($this->db)) {
  2658. $this->error('no database defined');
  2659. return false;
  2660. }
  2661. if (!isset ($this->tb)) {
  2662. $this->error('no table defined');
  2663. return false;
  2664. }
  2665. if ($this->dbh = @mysql_pconnect($this->hn, $this->un, $this->pw)) {
  2666. } else {
  2667. $this->error('could not connect to MySQL');
  2668. return false;
  2669. }
  2670. return true;
  2671. } /* }}} */
  2672. /*
  2673. * Database disconnection function
  2674. */
  2675. function disconnect() /* {{{ */
  2676. {
  2677. @mysql_close($this->dbh);
  2678. unset($this->dbh);
  2679. } /* }}} */
  2680. /*
  2681. * The workhorse
  2682. */
  2683. function execute() /* {{{ */
  2684. {
  2685. // DEBUG - uncomment to enable
  2686. /*
  2687. //phpinfo();
  2688. $this->print_get_vars();
  2689. $this->print_post_vars();
  2690. $this->print_vars();
  2691. echo "<pre>query opts:\n";
  2692. echo print_r($this->query_opts);
  2693. echo "</pre>\n";
  2694. echo "<pre>get vars:\n";
  2695. echo print_r($this->get_opts);
  2696. echo "</pre>\n";
  2697. */
  2698. // Let's do explicit quoting - it's safer
  2699. set_magic_quotes_runtime(0);
  2700. // Checking if language file inclusion was successful
  2701. if (! is_array($this->labels)) {
  2702. $this->error('could not locate language files', 'searched path: '.$this->dir['lang']);
  2703. return false;
  2704. }
  2705. // Database connection
  2706. if ($this->connect() == false) {
  2707. return false;
  2708. }
  2709. /*
  2710. * ======================================================================
  2711. * Pass 3: process any updates generated if the user has selected
  2712. * a save button during Pass 2
  2713. * ======================================================================
  2714. */
  2715. if ($this->saveadd == $this->labels['Save']) {
  2716. $this->add_enabled() && $this->do_add_record();
  2717. }
  2718. elseif ($this->moreadd == $this->labels['More']) {
  2719. $this->add_enabled() && $this->do_add_record();
  2720. $this->operation = $this->labels['Add']; // to force add operation
  2721. $this->recreate_fdd();
  2722. $this->recreate_displayed();
  2723. }
  2724. elseif ($this->savechange == $this->labels['Save']) {
  2725. $this->change_enabled() && $this->do_change_record();
  2726. }
  2727. elseif ($this->morechange == $this->labels['Apply']) {
  2728. $this->change_enabled() && $this->do_change_record();
  2729. $this->operation = $this->labels['Change']; // to force change operation
  2730. $this->recreate_fdd();
  2731. $this->recreate_displayed();
  2732. }
  2733. elseif ($this->savedelete == $this->labels['Delete']) {
  2734. $this->delete_enabled() && $this->do_delete_record();
  2735. }
  2736. /*
  2737. * ======================================================================
  2738. * Pass 2: display an input/edit/confirmation screen if the user has
  2739. * selected an editing button on Pass 1 through this page
  2740. * ======================================================================
  2741. */
  2742. if ($this->add_operation()
  2743. || $this->change_operation() || $this->delete_operation()
  2744. || $this->view_operation() || $this->copy_operation()) {
  2745. $this->display_record();
  2746. }
  2747. /*
  2748. * ======================================================================
  2749. * Pass 1 and Pass 3: display the MySQL table in a scrolling window on
  2750. * the screen (skip this step in 'Add More' mode)
  2751. * ======================================================================
  2752. */
  2753. else {
  2754. $this->list_table();
  2755. }
  2756. $this->disconnect();
  2757. if ($this->display['time'] && $this->timer != null) {
  2758. echo $this->timer->end(),' miliseconds';
  2759. }
  2760. } /* }}} */
  2761. /*
  2762. * Class constructor
  2763. */
  2764. function phpMyEdit($opts) /* {{{ */
  2765. {
  2766. // Set desirable error reporting level
  2767. $error_reporting = @error_reporting(E_ALL & ~E_NOTICE);
  2768. // Instance class variables
  2769. $this->hn = $opts['hn'];
  2770. $this->un = $opts['un'];
  2771. $this->pw = $opts['pw'];
  2772. $this->db = $opts['db'];
  2773. $this->tb = $opts['tb'];
  2774. $this->key = $opts['key'];
  2775. $this->key_type = $opts['key_type'];
  2776. $this->inc = $opts['inc'];
  2777. $this->options = $opts['options'];
  2778. $this->fdd = $opts['fdd'];
  2779. $this->multiple = intval($opts['multiple']);
  2780. $this->multiple <= 0 && $this->multiple = 2;
  2781. $this->filters = @$opts['filters'];
  2782. $this->triggers = @$opts['triggers'];
  2783. $this->notify = @$opts['notify'];
  2784. $this->logtable = @$opts['logtable'];
  2785. $this->page_name = @$opts['page_name'];
  2786. if (! isset($this->page_name)) {
  2787. $this->page_name = basename($this->get_server_var('PHP_SELF'));
  2788. isset($this->page_name) || $this->page_name = $this->tb;
  2789. }
  2790. $this->display['query'] = @$opts['display']['query'];
  2791. $this->display['sort'] = @$opts['display']['sort'];
  2792. $this->display['time'] = @$opts['display']['time'];
  2793. if ($this->display['time']) {
  2794. $this->timer = new phpMyEdit_timer();
  2795. }
  2796. $this->display['tabs'] = isset($opts['display']['tabs'])
  2797. ? $opts['display']['tabs'] : true;
  2798. $this->display['form'] = isset($opts['display']['form'])
  2799. ? $opts['display']['form'] : true;
  2800. // Creating directory variables
  2801. $this->dir['root'] = dirname(realpath(__FILE__))
  2802. . (strlen(dirname(realpath(__FILE__))) > 0 ? '/' : '');
  2803. $this->dir['lang'] = $this->dir['root'].'lang/';
  2804. // Creating URL variables
  2805. $this->url['images'] = 'images/';
  2806. isset($opts['url']['images']) && $this->url['images'] = $opts['url']['images'];
  2807. // CSS classes policy
  2808. $this->css = @$opts['css'];
  2809. !isset($this->css['separator']) && $this->css['separator'] = '-';
  2810. !isset($this->css['prefix']) && $this->css['prefix'] = 'pme';
  2811. !isset($this->css['page_type']) && $this->css['page_type'] = false;
  2812. !isset($this->css['position']) && $this->css['position'] = false;
  2813. !isset($this->css['divider']) && $this->css['divider'] = 2;
  2814. $this->css['divider'] = intval(@$this->css['divider']);
  2815. // Navigation
  2816. $this->navigation = @$opts['navigation'];
  2817. if (! $this->nav_buttons() && ! $this->nav_text_links() && ! $this->nav_graphic_links()) {
  2818. $this->navigation .= 'B'; // buttons are default
  2819. }
  2820. if (! $this->nav_up() && ! $this->nav_down()) {
  2821. $this->navigation .= 'D'; // down position is default
  2822. }
  2823. // Language labels (must go after navigation)
  2824. $this->labels = $this->make_language_labels(isset($opts['language'])
  2825. ? $opts['language'] : $this->get_server_var('HTTP_ACCEPT_LANGUAGE'));
  2826. // CGI variables
  2827. $this->cgi['append'] = @$opts['cgi']['append'];
  2828. $this->cgi['overwrite'] = @$opts['cgi']['overwrite'];
  2829. $this->cgi['persist'] = '';
  2830. if (@is_array($opts['cgi']['persist'])) {
  2831. foreach ($opts['cgi']['persist'] as $key => $val) {
  2832. $this->cgi['persist'] .= '&'.urlencode($key).'='.urlencode($val);
  2833. }
  2834. }
  2835. // Sorting variables
  2836. $this->sfn = $this->get_cgi_var('sfn');
  2837. isset($this->sfn) || $this->sfn = array();
  2838. is_array($this->sfn) || $this->sfn = array($this->sfn);
  2839. isset($opts['sort_field']) || $opts['sort_field'] = array();
  2840. is_array($opts['sort_field']) || $opts['sort_field'] = array($opts['sort_field']);
  2841. $this->sfn = array_merge($this->sfn, $opts['sort_field']);
  2842. #$this->sfn[] = '0'; // this last entry will be replaced in recreate_displayed()
  2843. // Form variables all around
  2844. $this->fl = intval($this->get_cgi_var('fl'));
  2845. $this->fm = intval($this->get_cgi_var('fm'));
  2846. $this->qfn = $this->get_cgi_var('qfn');
  2847. $this->sw = $this->get_cgi_var('sw');
  2848. $this->rec = $this->get_cgi_var('rec', '');
  2849. $this->navop = $this->get_cgi_var('navop');
  2850. if (($this->navfm = $this->get_cgi_var('navfmup', $this->fm)) != $this->fm) {
  2851. $this->navop = $this->labels['Go to'];
  2852. } else if (($this->navfm = $this->get_cgi_var('navfmdown', $this->navfm)) != $this->fm) {
  2853. $this->navop = $this->labels['Go to'];
  2854. }
  2855. $this->operation = $this->get_cgi_var('operation');
  2856. $this->saveadd = $this->get_cgi_var('saveadd');
  2857. $this->moreadd = $this->get_cgi_var('moreadd');
  2858. $this->canceladd = $this->get_cgi_var('canceladd');
  2859. $this->savechange = $this->get_cgi_var('savechange');
  2860. $this->morechange = $this->get_cgi_var('morechange');
  2861. $this->cancelchange = $this->get_cgi_var('cancelchange');
  2862. $this->savedelete = $this->get_cgi_var('savedelete');
  2863. $this->canceldelete = $this->get_cgi_var('canceldelete');
  2864. $this->cancelview = $this->get_cgi_var('cancelview');
  2865. // Filter setting
  2866. if (isset($this->sw)) {
  2867. $this->sw == $this->labels['Search'] && $this->fl = 1;
  2868. $this->sw == $this->labels['Hide'] && $this->fl = 0;
  2869. //$this->sw == $this->labels['Clear'] && $this->fl = 0;
  2870. }
  2871. // TAB names
  2872. $this->tabs = array();
  2873. // Setting key_delim according to key_type
  2874. if ($this->key_type == 'real') {
  2875. /* If 'real' key_type does not work,
  2876. try change MySQL datatype from float to double */
  2877. $this->rec = doubleval($this->rec);
  2878. $this->key_delim = '';
  2879. } elseif ($this->key_type == 'int') {
  2880. $this->rec = intval($this->rec);
  2881. $this->key_delim = '';
  2882. } else {
  2883. $this->key_delim = '"';
  2884. // $this->rec remains unmodified
  2885. }
  2886. // Specific $fdd modifications depending on performed action
  2887. $this->recreate_fdd();
  2888. // Extract SQL Field Names and number of fields
  2889. $this->recreate_displayed();
  2890. // Gathering query options
  2891. $this->gather_query_opts();
  2892. // Call to action
  2893. !isset($opts['execute']) && $opts['execute'] = 1;
  2894. $opts['execute'] && $this->execute();
  2895. // Restore original error reporting level
  2896. @error_reporting($error_reporting);
  2897. } /* }}} */
  2898. }
  2899. /* Modeline for ViM {{{
  2900. * vim:set ts=4:
  2901. * vim600:fdm=marker fdl=0 fdc=0:
  2902. * }}} */
  2903. ?>