phpMyEdit2.class.php 106 KB

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