Fix driver display in order form.
[readifood.git] / lib / order.php
1 <?php
2
3   if (isset($_POST['show_add_order'])) {
4     $area_id = $_POST['area_id'];
5     show_new_order_form($area_id);
6   }
7   else if (isset($_POST['add_order'])) {
8     $id = add_order();
9     if ($id !== false) {
10       echo "<p>Order placed.</p>\n";
11       $parameters = array("id", $id);
12     }
13   }
14   else if (isset($_POST['update_order'])) {
15     list($ignored, $id, $args) = parse_parameters($parameters);
16     $q = new OrderQuery;
17     $order = $q->findOneById($id);
18     if ($order) {
19       if (update_order($order) !== false) {
20         echo "<p>Updated order.</p>\n";
21         $parameters = array("id", $order->getId());
22       }
23     }
24     else {
25       echo "<p>No such contact!</p>\n";
26     }
27   }
28   else if ($_POST['show_in_area']) {
29     $q = new AreaQuery;
30     $area = $q->findOneById($_POST['area_id']);
31     header(sprintf("Location: http%s://%s/%s/in/area/%s/%d%s", ($_SERVER['HTTPS']) ? "s" : "", $_SERVER['HTTP_HOST'], $module, urlencode($area->getName()), $_POST['area_id'], get_order_state_uri(get_order_state_mask())));
32     exit;
33   }
34   else if ($_POST['show_in_city']) {
35     $q = new CityQuery;
36     $city = $q->findOneById($_POST['city_id']);
37     header(sprintf("Location: http%s://%s/%s/in/city/%s/%d%s", ($_SERVER['HTTPS']) ? "s" : "", $_SERVER['HTTP_HOST'], $module, urlencode($city->getName()), $_POST['city_id'], get_order_state_uri(get_order_state_mask())));
38     exit;
39   }
40
41   function show_orders($offset, $per_page, $requester_ids = null, $beneficiary_ids = null, $state_mask = null) {
42     /* XXX: Use Propel methods. */
43     if (isset($state_mask)) {
44       $order_ids = array();
45       $dbh = Propel::getConnection();
46       $sth = $dbh->prepare("select * from OrderState o where updated=(select max(updated) from OrderState where order_id=o.order_id) and state & $state_mask");
47       $sth->execute();
48       $order_states = OrderStatePeer::populateObjects($sth);
49       foreach ($order_states as $order_state) $order_ids[] = $order_state->getOrderId();
50     }
51     $q = new OrderQuery;
52     if (isset($requester_ids)) $q->filterByRequesterId($requester_ids);
53     if (isset($beneficiary_ids)) $q->filterByBeneficiaryId($beneficiary_ids);
54     # XXX: Doesn't work.
55     #if (isset($state_mask)) $q->useOrderStateQuery()->addSelectQuery($latest_state, 'latestState')->where("order_id=latestState.order_id")->where("state & $state_mask")->endUse();
56     if (isset($state_mask)) $q->filterById($order_ids);
57     $orders = $q->orderByDate()->find();
58     if (count($orders)) {
59       foreach ($orders as $order) {
60         echo "<br>\nOrder " . $order->getStrongLink($order->getId()) . ": " . get_order_displayname($order);
61         if (check_admin(1)) {
62           echo " " . $order->getDeleteLink();
63         }
64
65         /* XXX: Should pull from query. */
66         $q = new ContactQuery;
67         $contact = $q->findOneById($order->getBeneficiaryId());
68         if ($contact) {
69           echo " for " . $contact->getLink();
70           $area = get_contact_area($contact);
71           if ($area) echo " in " . $area->getLink();
72         }
73
74         if ($order->getHubId()) {
75           $q = new HubQuery;
76           $hub = $q->findOneById($order->getHubId());
77           if ($hub) echo " to hub " . $hub->getLink();
78           $area = get_hub_area($hub);
79           if ($area) echo " in " . $area->getLink();
80         }
81       }
82     }
83     else echo " none";
84   }
85
86   function show_city_orders($offset, $per_page, $city_name, $city_id = null, $state_mask = null) {
87     if (isset($city_id)) $city = get_city_by_id($city_id);
88     else if ($city_name) $city = get_city_by_name($city_name);
89     if ($city) {
90       $contacts = get_city_contacts($city->getId(), $GLOBALS['ROLE_BENEFICIARY']);
91       $beneficiary_ids = array();
92       foreach ($contacts as $contact) $beneficiary_ids[] = $contact->getId();
93
94       echo "<p>Orders in city " . $city->getLink(get_city_displayname($city)) . ":";
95       return show_orders($offset, $per_page, null, $beneficiary_ids, $state_mask);
96     }
97     else echo "<p>No such city!</p>\n";
98   }
99
100   function show_requester_orders($offset, $per_page, $contact_name, $contact_id = null, $state_mask = null) {
101     if (isset($contact_id)) $contact = get_contact_by_id($contact_id);
102     else if ($contact_name) $contact = get_contact_by_name($contact_name);
103     if ($contact) {
104       echo "<p>Orders from referrer " . $contact->getLink() . ":";
105       return show_orders($offset, $per_page, $contact->getId(), null, $state_mask);
106     }
107     else echo "<p>No such contact!</p>\n";
108   }
109
110   function show_beneficiary_orders($offset, $per_page, $contact_name, $contact_id = null, $state_mask = null) {
111     if (isset($contact_id)) $contact = get_contact_by_id($contact_id);
112     else if ($contact_name) $contact = get_contact_by_name($contact_name);
113     if ($contact) {
114       echo "<p>Orders to beneficiary " . $contact->getLink() . ":";
115       return show_orders($offset, $per_page, null, $contact->getId(), $state_mask);
116     }
117     else echo "<p>No such contact!</p>\n";
118   }
119
120   function show_area_orders($offset, $per_page, $area_name, $area_id = null, $state_mask = null) {
121     if (isset($area_id)) $area = get_area_by_id($area_id);
122     else if ($area_name) $area = get_area_by_name($area_name);
123     if ($area) {
124       $contacts = get_area_contacts($area->getId(), $GLOBALS['ROLE_BENEFICIARY']);
125       $contact_ids = array();
126       foreach ($contacts as $contact) $contact_ids[] = $contact->getId();
127
128       echo "<p>Orders in area " . $area->getLink() . ":";
129       return show_orders($offset, $per_page, null, $contact_ids, $state_mask);
130     }
131     else echo "<p>No such area!</p>\n";
132   }
133
134   function show_order_state_form($state_mask = null) {
135     global $states, $all_states;
136
137     if (is_null($state_mask)) $state_mask = $all_states;
138
139     echo "<p>Restrict to order states:\n";
140     for ($i = 0; $i < count($states); $i++) {
141       echo " <input type=\"checkbox\" name=\"state_$i\"";
142       if ($state_mask & (1 << $i)) echo " checked";
143       echo ">$states[$i]\n";
144     }
145     echo "</p>\n";
146   }
147
148   function get_order_state_mask($string = null) {
149     global $states, $all_states;
150
151     $mask = 0;
152
153     if (isset($string)) {
154       $selected = explode("+", $string);
155       for ($i = 0; $i < count($states); $i++) {
156         if (in_array($states[$i], $selected)) $mask |= (1 << $i);
157       }
158     }
159     else {
160       for ($i = 0; $i < count($states); $i++) {
161         if ($_POST['state_' . $i] == "on") $mask |= (1 << $i);
162       }
163     }
164
165     if (! $mask) $mask = $all_states;
166     return $mask;
167   }
168
169   function get_order_state_string($mask) {
170     global $states;
171
172     $selected = array();
173
174     for ($i = 0; $i < count($states); $i++) {
175       if ($mask & (1 << $i)) $selected[] = $states[$i];
176     }
177
178     return implode("+", $selected);
179   }
180
181   function get_order_state_uri($mask) {
182     global $all_states;
183
184     if (is_null($mask)) return "";
185     if ($mask == $all_states) return "";
186
187     return "/state/" . get_order_state_string($mask);
188   }
189
190   function show_order_areas_form($city_id = null) {
191     $areas = get_city_areas($city_id);
192     if (! count($areas)) {
193       echo "<p>No <a href=\"/area\">areas</a>!</p>\n";
194       return;
195     }
196
197     echo "<p>Show orders in area\n";
198     echo "<select name=\"area_id\">\n";
199     foreach ($areas as $area) {
200       option("area_id", $area->getId(), get_area_displayname($area));
201     }
202     echo "</select>\n";
203     echo "<input type=\"submit\" name=\"show_in_area\" value=\"Show\">\n";
204   }
205
206   function show_order_cities_form($city_id = null) {
207     $q = new CityQuery;
208     $cities = $q->orderByName()->find();
209
210     if (! count($cities)) {
211       echo "<p>No <a href=\"/city\">cities</a>!</p>\n";
212       return;
213     }
214
215     echo "<p>Show orders in city\n";
216     echo "<select name=\"city_id\">\n";
217     foreach ($cities as $city) {
218       option("city_id", $city->getId(), get_city_displayname($city), $city_id);
219     }
220     echo "</select>\n";
221     echo "<input type=\"submit\" name=\"show_in_city\" value=\"Show\">\n";
222   }
223
224   function show_order_forms($city_id, $state_mask) {
225     echo "<form method=\"POST\" action=\"" . $_SERVER['REQUEST_URI'] . "\">\n";
226     show_order_state_form($state_mask);
227     show_order_areas_form($city_id);
228     show_order_cities_form($city_id);
229     echo "</form>\n";
230   }
231
232   function show_order_form($order = null, $area_id = null) {
233     global $states;
234
235     if ($order) {
236       $q = new OrderStateQuery;
237       $order_state = $q->filterByOrderId($order->getId())->orderByUpdated('desc')->limit(1)->findOne();
238       if ($order_state) {
239         $state = $order_state->getState();
240         $driver_id = $order_state->getDriverId();
241       }
242     }
243     else $order = new Order;
244
245
246     /* Date. */
247     echo "<tr>\n";
248     echo "  <td>Delivery</td>\n";
249     /* XXX: Find suitable dates from area. */
250     echo "  <td>";
251     show_date_form("date", $order->getDate());
252     if (! $order->getDate()) {
253       echo " and recur for <select name=\"recurrence\">\n";
254       for ($i = 0; $i < 4; $i++) option("recurrence", $i, $i);
255       echo "</select> weeks";
256     }
257     echo "</td>\n";
258     echo "</tr>\n";
259
260     /* Referrer. */
261     echo "<tr>\n";
262     echo "  <td>Referrer</td>\n";
263     echo "  <td><select name=\"requester_id\">\n";
264     option("requester_id", null, "");
265     $contacts = get_area_requesters();
266     foreach ($contacts as $contact) {
267       option("requester_id", $contact->getId(), $contact->getDisplayname(), $order->getRequesterId());
268     }
269     echo "</select></td>\n";
270     echo "</tr>\n";
271
272     /* Beneficiary. */
273     echo "<tr>\n";
274     echo "  <td>Beneficiary</td>\n";
275     echo "  <td><select name=\"beneficiary_id\">\n";
276     option("beneficiary_id", null, "");
277     $contacts = get_area_beneficiaries($area_id);
278     foreach ($contacts as $contact) {
279       option("beneficiary_id", $contact->getId(), $contact->getDisplayname(), $order->getBeneficiaryId());
280     }
281     echo "</select></td>\n";
282     echo "</tr>\n";
283
284     /* Hub. */
285     echo "<tr>\n";
286     echo "  <td>Hub</td>\n";
287     echo "  <td><select name=\"hub_id\">\n";
288     option("hub_id", null, "");
289     $hubs = get_area_hubs();
290     foreach ($hubs as $hub) {
291       option("hub_id", $hub->getId(), $hub->getDisplayname(), $order->getHubId());
292     }
293     echo "</select></td>\n";
294     echo "</tr>\n";
295
296     /* Quantity. */
297     echo "<tr>\n";
298     echo "  <td>Quantity (kg)</td>\n";
299     echo "  <td>"; input("quantity", sprintf("%0.2f", $order->getQuantity() / 1000)); echo "</td>\n";
300     echo "</tr>\n";
301
302     /* Driver. */
303     echo "<tr>\n";
304     echo "  <td>Driver</td>\n";
305     $contacts = get_city_drivers();
306     if (count($contacts)) {
307       echo "  <td><select name=\"driver_id\">\n";
308       option("driver_id", null, "");
309       foreach ($contacts as $contact) {
310         option("driver_id", $contact->getId(), $contact->getDisplayname(), $driver_id);
311       }
312       echo "</select></td>\n";
313     }
314     else echo "  <td>No drivers!</td>\n";
315     echo "</tr>\n";
316
317     /* State. */
318     if ($order->getId()) {
319       echo "<tr>\n";
320       echo "  <td>State</td>\n";
321       echo "  <td><select name=\"state\">\n";
322       for ($i = 0; $i < count($states); $i++) {
323         option("state", $i << 1, ucfirst($states[$i]), $state);
324       }
325       echo "</select></td>\n";
326       echo "</tr>\n";
327     }
328   }
329
330   function show_new_order_form($area_id = null) {
331     if (! check_admin(1)) return;
332
333     $area = get_area_by_id($area_id);
334     if (! count($area)) {
335       echo "<p>No such <a href=\"/area\">area</a>!</p>\n";
336       return;
337     }
338
339     echo "<form method=\"POST\" action=\"" . $_SERVER['REQUEST_URI'] . "\">\n";
340     echo "<p>Place an order:</p>\n";
341
342     echo "<table>\n";
343     show_order_form(null, $area_id);
344
345     echo "<tr>\n";
346     echo "  <td colspan=2>"; submit("add_order", "Order"); echo "</td></tr>\n";
347     echo "</tr>\n";
348     echo "</table>\n";
349     echo "</form>\n";
350   }
351
352   function show_add_new_order_form() {
353     if (! check_admin(1)) return;
354
355     /* We intentionally hide areas with no contacts. */
356     $areas = get_city_areas_with_contacts(null, $GLOBALS['ROLE_BENEFICIARY']);
357     if (! count($areas)) {
358       echo "<p>Can't place any orders until at least one <a href=\"/area\">area</a> has a <a href=\"/contact\">contact</a>!</p>\n";
359       return;
360     }
361
362     echo "<form method=\"POST\" action=\"" . $_SERVER['REQUEST_URI'] . "\">\n";
363     echo "<p>Place an order in <select name=\"area_id\">\n";
364     foreach ($areas as $area) {
365       option("area_id", $area->getId(), get_area_displayname($area));
366     }
367     echo "</select>";
368     submit("show_add_order", "Proceed");
369     echo "</p>\n";
370     echo "</form>\n";
371   }
372
373   function update_order(&$order, $new = false) {
374     global $user_id;
375
376     #$date = ymd_to_iso8601("date");
377     $date = $_POST['date'];
378     $requester_id = $_POST['requester_id'];
379     $beneficiary_id = $_POST['beneficiary_id'];
380     $hub_id = $_POST['hub_id'];
381     $quantity = $_POST['quantity'];
382     $driver_id = $_POST['driver_id'];
383     if (! $driver_id) $driver_id = null;
384     $state = $_POST['state'];
385     if (! $state) $state = $GLOBALS['STATE_PLACED'];
386
387     if ($date) {
388       list($y, $m, $d) = explode('-', $date);
389       $then = mktime(0, 0, 0, $m, $d, $y);
390     }
391     else $then = time();
392     /* XXX: check date */
393
394     $requester = get_contact_by_id($requester_id);
395     if (! $requester) {
396       echo "<p>Invalid referrer!</p>\n";
397       return false;
398     }
399
400     $beneficiary = get_contact_by_id($beneficiary_id);
401     if (! $beneficiary) {
402       echo "<p>Invalid beneficiary!</p>\n";
403       return false;
404     }
405
406     if ($hub_id) {
407       $hub = get_hub_by_id($hub_id);
408       if (! $hub) {
409         echo "<p>Invalid hub!</p>\n";
410         return false;
411       }
412     }
413     else $hub_id = null;
414
415     if (! is_numeric($quantity)) {
416       echo "<p>Invalid quantity!</p>\n";
417       return false;
418     }
419
420     if ($new && isset($_POST['recurrence'])) $recurrence = $_POST['recurrence'];
421     if (! $recurrence) $recurrence = 0;
422
423     $now = time();
424     for ($i = 0; $i <= $recurrence; $i++) {
425       if ($i) {
426         echo "<p>Creating recurrence $i.</p>\n";
427         $order = new Order;
428       }
429
430       $order->setDate($then + 7 * 86400 * $i);
431       $order->setRequesterId($requester_id);
432       $order->setBeneficiaryId($beneficiary_id);
433       $order->setHubId($hub_id);
434       $order->setQuantity($quantity * 1000);
435
436       /* XXX: begin/commit */
437       try {
438         $order->save();
439
440         $order_state = new OrderState;
441         $order_state->setUpdated($now);
442         $order_state->setOrderId($order->getId());
443         $order_state->setUserId($user_id);
444         $order_state->setDriverId($driver_id);
445         $order_state->setState($state);
446
447         $order_state->save();
448       }
449       catch (Exception $e) {
450         if ($new) echo "<p>Error placing order.</p>\n";
451         else echo "<p>Error updating order.</p>\n";
452         echo "<p>" . $e->getMessage() . "</p>\n";
453         return false;
454       }
455     }
456
457     return true;
458   }
459
460   function add_order() {
461     if (! check_admin(1, "place an order")) return;
462
463     $order = new Order;
464     if (! update_order($order, true)) return false;
465     return $order->getId();
466   }
467
468   function delete_order($id = null) {
469     if (! check_admin(1, "delete an order")) return;
470
471     if (isset($id)) $order = get_order_by_id($id);
472     if (! $order) return false;
473
474     try {
475       $q = new OrderStateQuery;
476       $order_states = $q->filterByOrderId($id)->find();
477       foreach ($order_states as $order_state) $order_state->delete();
478       $order->delete();
479       echo "<p>Deleted order.</p>\n";
480     }
481     catch (Exception $e) {
482       echo "<p>Error deleting order $id!</p>\n";
483       return false;
484     }
485
486     return true;
487   }
488
489   function show_order_history($id) {
490     global $states;
491
492     $q = new OrderStateQuery();
493     $order_states = $q->filterByOrderId($id)->orderById()->find();
494
495     if (! count($order_states)) return;
496
497     echo "<h3>Order history</h3>\n";
498     foreach ($order_states as $order_state) {
499       $date = $order_state->getUpdated();
500
501       $user = get_contact_by_id($order_state->getUserId());
502       if ($user) $username = $user->getDisplayname();
503       else $username = "unknown user";
504
505       $driver_id = $order_state->getDriverId();
506       if ($driver_id) $driver = get_contact_by_id($driver_id);
507       else $driver = null;
508
509       /* XXX */
510       $state = $order_state->getState();
511       for ($i = 0; $i < count($states); $i++) {
512         if ((1 << $i) == $state) {
513           $state = $states[$i];
514           break;
515         }
516       }
517       #$state = $states[$order_state->getState()];
518       echo "<p><strong>$username</strong> changed order to state <strong>$state</strong>";
519       if ($driver) echo " for driver " . $driver->getDisplayname();
520       echo " on $date.</p>\n";
521     }
522   }
523
524   function show_order(&$id = null) {
525     if (isset($id)) $order = get_order_by_id($id);
526     if (! $order) return;
527
528     echo "<form method=\"POST\" action=\"" . $_SERVER['REQUEST_URI'] . "\">\n";
529     echo "<p>Order: <span class=\"strong\">" . $order->getId() . "</span>";
530     if (check_admin(1)) {
531       echo " " . $order->getDeleteLink();
532     }
533     echo ": ";
534     echo "\n</p>";
535
536     echo "<table>\n";
537     show_order_form($order);
538
539     if (check_admin(1)) {
540       echo "<tr>\n";
541       echo "  <td colspan=2>";
542       submit("update_order", "Update");
543       echo "</td>\n";
544       echo "</tr>\n";
545     }
546
547     echo "</table>\n";
548     echo "</form>\n";
549
550     show_order_history($order->getId());
551   }
552
553   $state_mask = null;
554   if (count($parameters)) {
555     for ($i = 1; $i < count($parameters); $i++) {
556       if ($parameters[$i] == "state") {
557         /* /order/state/placed+picked */
558         $state_mask = get_order_state_mask($parameters[$i + 1]);
559       }
560     }
561
562     if ($parameters[0] == "in") {
563       /* /order/in/area/Romsey+Town/1 */
564       switch ($parameters[1]) {
565       case "area":
566         case "area":
567           $area_id = $parameters[3];
568           $_POST['area_id'] = $area_id;
569           $q = new AreaQuery;
570           $area = $q->findOneById($area_id);
571           $city = get_area_city($area);
572           if ($city) $city_id = $city->getId();
573           show_area_orders(0, 10, $parameters[2], $area_id, $state_mask);
574         break;
575
576         case "city":
577           $city_id = $parameters[3];
578           $_POST['city_id'] = $city_id;
579           $q = new CityQuery;
580           $city = $q->findOneById($city_id);
581           show_city_orders(0, 10, $parameters[2], $city_id, $state_mask);
582         break;
583       }
584     }
585     else if ($parameters[0] == "from") {
586       /* /order/from/referrer/Iain+Patterson/4 */
587       switch ($parameters[1]) {
588         case "referrer":
589           $contact_id = $parameters[3];
590           $q = new ContactQuery;
591           $contact = $q->findOneById($contact_id);
592           show_requester_orders(0, 10, $parameters[2], $contact_id, $state_mask);
593         break;
594       }
595     }
596     else if ($parameters[0] == "to") {
597       /* /order/to/beneficiary/Cambridge+Community+Church/1 */
598       switch ($parameters[1]) {
599         case "beneficiary":
600           $contact_id = $parameters[3];
601           $q = new ContactQuery;
602           $hub = $q->findOneById($contact_id);
603           show_beneficiary_orders(0, 10, $parameters[2], $contact_id, $state_mask);
604         break;
605       }
606     }
607   }
608   list($ignored, $id, $args) = parse_parameters($parameters);
609   //echo "<p>$name($id) " . print_r($args, true) . "</p>\n";
610   if (count($args)) {
611     switch ($args[0]) {
612       case "delete":
613         delete_order($id);
614       break;
615     }
616   }
617   else if (isset($id)) show_order($id);
618   else if ($state_mask) show_orders(0, 10, null, null, $state_mask);
619   else {
620     /* XXX: Shown after adding. */
621     show_order_forms($city_id, $state_mask);
622     show_add_new_order_form($city_id);
623   }
624
625   if (count($parameters)) {
626     show_order_forms($city_id, $state_mask);
627   }
628
629
630 ?>