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