2 * $Id: roomlist.c 7751 2009-08-28 21:13:28Z dothebart $
3 * room listings and filters.
10 void DeleteFloor(void *vFloor)
13 Floor = (floor*) vFloor;
14 FreeStrBuf(&Floor->Name);
18 int SortFloorsByNameOrder(const void *vfloor1, const void *vfloor2)
20 floor *f1 = (floor*) GetSearchPayload(vfloor1);
21 floor *f2 = (floor*) GetSearchPayload(vfloor2);
23 /* prefer My floor over alpabetical sort */
24 if (f1->ID == VIRTUAL_MY_FLOOR)
26 if (f2->ID == VIRTUAL_MY_FLOOR)
29 return strcmp(ChrPtr(f1->Name), ChrPtr(f2->Name));
32 HashList *GetFloorListHash(StrBuf *Target, WCTemplputParams *TP) {
46 if (WCC->Floors != NULL)
48 WCC->Floors = floors = NewHash(1, NULL);
51 Floor = malloc(sizeof(floor));
52 Floor->ID = VIRTUAL_MY_FLOOR;
53 Floor->Name = NewStrBufPlain(_("My Folders"), -1);
56 Put(floors, IKEY(Floor->ID), Floor, DeleteFloor);
58 serv_puts("LFLR"); /* get floors */
59 StrBufTCP_read_line(Buf, &WC->serv_sock, 0, &Err); /* '100', we hope */
60 if (GetServerStatus(Buf, NULL) == 1)
62 while(StrBufTCP_read_line(Buf, &WC->serv_sock, 0, &Err), strcmp(ChrPtr(Buf), "000"))
67 Floor = malloc(sizeof(floor));
68 Floor->ID = StrBufExtractNext_long(Buf, &Pos, '|');
69 Floor->Name = NewStrBufPlain(NULL, StrLength(Buf));
70 StrBufExtract_NextToken(Floor->Name, Buf, &Pos, '|');
71 Floor->NRooms = StrBufExtractNext_long(Buf, &Pos, '|');
73 Put(floors, IKEY(Floor->ID), Floor, DeleteFloor);
78 /* now lets pre-sort them alphabeticaly. */
80 SortByPayload(floors, SortFloorsByNameOrder);
81 it = GetNewHashPos(floors, 0);
82 while ( GetNextHashPos(floors, it, &HKLen, &HashKey, &vFloor))
83 ((floor*) vFloor)->AlphaN = i++;
85 SortByHashKeyStr(floors);
90 void tmplput_FLOOR_ID(StrBuf *Target, WCTemplputParams *TP)
92 floor *Floor = (floor *)(TP->Context);
94 StrBufAppendPrintf(Target, "%d", Floor->ID);
97 void tmplput_FLOOR_NAME(StrBuf *Target, WCTemplputParams *TP)
99 floor *Floor = (floor *)(TP->Context);
101 StrBufAppendTemplate(Target, TP, Floor->Name, 0);
104 void tmplput_FLOOR_NROOMS(StrBuf *Target, WCTemplputParams *TP)
106 floor *Floor = (floor *)(TP->Context);
108 StrBufAppendPrintf(Target, "%d", Floor->NRooms);
110 HashList *GetRoomListHashLKRA(StrBuf *Target, WCTemplputParams *TP)
114 if (WCC->Floors == NULL)
115 GetFloorListHash(Target, TP);
117 return GetRoomListHash(Target, TP);
120 void DeleteFolder(void *vFolder)
124 room = (folder*) vFolder;
126 FreeStrBuf(&room->name);
127 ////FreeStrBuf(&room->ACL);
129 //// FreeStrBuf(&room->room);
131 if (room->RoomNameParts != NULL)
133 for (i=0; i < room->nRoomNameParts; i++)
134 FreeStrBuf(&room->RoomNameParts[i]);
135 free(room->RoomNameParts);
141 HashList *GetRoomListHash(StrBuf *Target, WCTemplputParams *TP)
151 WCTemplputParams SubTP;
154 rooms = NewHash(1, NULL);
155 StrBufTCP_read_line(Buf, &WC->serv_sock, 0, &Err);
156 if (GetServerStatus(Buf, NULL) == 1)
158 while(StrBufTCP_read_line(Buf, &WC->serv_sock, 0, &Err),
159 strcmp(ChrPtr(Buf), "000"))
163 room = (folder*) malloc (sizeof(folder));
164 memset(room, 0, sizeof(folder));
166 /* Load the base data from the server reply */
167 room->name = NewStrBufPlain(NULL, StrLength(Buf));
168 StrBufExtract_NextToken(room->name, Buf, &Pos, '|');
170 room->QRFlags = StrBufExtractNext_long(Buf, &Pos, '|');
171 room->floorid = StrBufExtractNext_long(Buf, &Pos, '|');
172 room->listorder = StrBufExtractNext_long(Buf, &Pos, '|');
173 room->QRFlags2 = StrBufExtractNext_long(Buf, &Pos, '|');
175 room->RAFlags = StrBufExtractNext_long(Buf, &Pos, '|');
179 room->ACL = NewStrBufPlain(NULL, StrLength(Buf));
180 StrBufExtract_NextToken(room->ACL, Buf, &Pos, '|');
183 room->view = StrBufExtractNext_long(Buf, &Pos, '|');
184 room->defview = StrBufExtractNext_long(Buf, &Pos, '|');
185 room->lastchange = StrBufExtractNext_long(Buf, &Pos, '|');
187 /* Evaluate the Server sent data for later use */
188 /* find out, whether we are in a sub-room */
189 room->nRoomNameParts = StrBufNum_tokens(room->name, '\\');
190 if (room->nRoomNameParts > 1)
195 room->RoomNameParts = malloc(sizeof(StrBuf*) * (room->nRoomNameParts + 1));
196 memset(room->RoomNameParts, 0, sizeof(StrBuf*) * (room->nRoomNameParts + 1));
197 for (i=0; i < room->nRoomNameParts; i++)
199 room->RoomNameParts[i] = NewStrBuf();
200 StrBufExtract_NextToken(room->RoomNameParts[i],
201 room->name, &Pos, '\\');
205 /* Private mailboxes on the main floor get remapped to the personal folder */
206 if ((room->QRFlags & QR_MAILBOX) &&
207 (room->floorid == 0))
208 room->floorid = VIRTUAL_MY_FLOOR;
209 /* get a pointer to the floor we're on: */
210 GetHash(WCC->Floors, IKEY(room->floorid), &vFloor);
211 room->Floor = (const floor*) vFloor;
215 /* now we know everything, remember it... */
216 Put(rooms, SKEY(room->name), room, DeleteFolder);
219 /// SortByHashKey(rooms, 1);
221 SubTP.Filter.ContextType = CTX_ROOMS;
222 SortIt = RetrieveSort(&SubTP, NULL, 0, HKEY("fileunsorted"), 0);
224 SortByPayload(rooms, SortIt);
226 SortByPayload(rooms, SortRoomsByListOrder);
231 /** Unused function that orders rooms by the listorder flag */
232 int SortRoomsByListOrder(const void *room1, const void *room2)
234 folder *r1 = (folder*) GetSearchPayload(room1);
235 folder *r2 = (folder*) GetSearchPayload(room2);
237 if (r1->listorder == r2->listorder) return 0;
238 if (r1->listorder > r2->listorder) return 1;
242 int CompareRoomListByFloorRoomPrivFirst(const void *room1, const void *room2)
244 folder *r1 = (folder*) GetSearchPayload(room1);
245 folder *r2 = (folder*) GetSearchPayload(room2);
247 if ((r1->Floor == NULL) ||
252 * are we on the same floor? else sort by floor.
254 if (r1->Floor != r2->Floor)
257 * the private rooms are first in any case.
259 if (r1->Floor->ID == VIRTUAL_MY_FLOOR)
261 if (r2->Floor->ID == VIRTUAL_MY_FLOOR)
264 * else decide alpaheticaly by floorname
266 return (r1->Floor->AlphaN > r2->Floor->AlphaN)? 1 : -1;
270 * if we have different levels of subdirectories,
271 * we want the toplevel to be first, regardless of sort
274 if (((r1->nRoomNameParts > 1) ||
275 (r2->nRoomNameParts > 1) )&&
276 (r1->nRoomNameParts != r2->nRoomNameParts))
279 int nparts = (r1->nRoomNameParts > r2->nRoomNameParts)?
280 r2->nRoomNameParts : r1->nRoomNameParts;
282 for (i=0; i < nparts; i++)
284 ret = strcmp (ChrPtr(r1->name),
287 * Deltas in common parts? exit here.
294 * who's a subdirectory of whom?
296 if (r1->nRoomNameParts > r2->nRoomNameParts)
304 * else just sort alphabeticaly.
306 return strcmp (ChrPtr(r1->name),
310 int CompareRoomListByFloorRoomPrivFirstRev(const void *room1, const void *room2)
312 folder *r1 = (folder*) GetSearchPayload(room1);
313 folder *r2 = (folder*) GetSearchPayload(room2);
315 if ((r1->Floor == NULL) ||
320 * are we on the same floor? else sort by floor.
322 if (r2->Floor != r1->Floor)
325 * the private rooms are first in any case.
327 if (r1->Floor->ID == VIRTUAL_MY_FLOOR)
329 if (r2->Floor->ID == VIRTUAL_MY_FLOOR)
332 * else decide alpaheticaly by floorname
335 return (r1->Floor->AlphaN < r2->Floor->AlphaN)? 1 : -1;
339 * if we have different levels of subdirectories,
340 * we want the toplevel to be first, regardless of sort
343 if (((r1->nRoomNameParts > 1) ||
344 (r2->nRoomNameParts > 1) )&&
345 (r1->nRoomNameParts != r2->nRoomNameParts))
348 int nparts = (r1->nRoomNameParts > r2->nRoomNameParts)?
349 r2->nRoomNameParts : r1->nRoomNameParts;
351 for (i=0; i < nparts; i++)
354 * special cases if one room is top-level...
356 if (r2->nRoomNameParts == 1)
357 ret = strcmp (ChrPtr(r2->name),
358 ChrPtr(r1->RoomNameParts[i]));
359 else if (r1->nRoomNameParts == 1)
360 ret = strcmp (ChrPtr(r2->RoomNameParts[i]),
363 ret = strcmp (ChrPtr(r2->RoomNameParts[i]),
364 ChrPtr(r1->RoomNameParts[i]));
366 * Deltas in common parts? exit here.
373 * who's a subdirectory of whom?
375 if (r1->nRoomNameParts > r2->nRoomNameParts)
381 return strcmp (ChrPtr(r2->name),
385 int GroupchangeRoomListByFloorRoomPrivFirst(const void *room1, const void *room2)
387 folder *r1 = (folder*) room1;
388 folder *r2 = (folder*) room2;
391 if ((r1->Floor == NULL) ||
395 if (r1->Floor == r2->Floor)
400 static int columns = 3;
401 int boxes_per_column = 0;
404 nf = GetCount(WCC->Floors);
405 while (nf % columns != 0) ++nf;
406 boxes_per_column = (nf / columns);
407 if (boxes_per_column < 1)
408 boxes_per_column = 1;
409 if (r1->Floor->AlphaN % boxes_per_column == 0)
413 /// wprintf("</td><td valign=top>\n");
422 void tmplput_ROOM_NAME(StrBuf *Target, WCTemplputParams *TP)
424 folder *Folder = (folder *)(TP->Context);
426 StrBufAppendTemplate(Target, TP, Folder->name, 0);
428 void tmplput_ROOM_BASENAME(StrBuf *Target, WCTemplputParams *TP)
430 folder *room = (folder *)(TP->Context);
432 if (room->nRoomNameParts > 1)
433 StrBufAppendTemplate(Target, TP,
434 room->RoomNameParts[room->nRoomNameParts - 1], 0);
436 StrBufAppendTemplate(Target, TP, room->name, 0);
438 void tmplput_ROOM_LEVEL_N_TIMES(StrBuf *Target, WCTemplputParams *TP)
440 folder *room = (folder *)(TP->Context);
442 const char *AppendMe;
446 if (room->nRoomNameParts > 1)
448 GetTemplateTokenString(Target, TP, 0, &AppendMe, &AppendMeLen);
449 for (i = 0; i < room->nRoomNameParts; i++)
450 StrBufAppendBufPlain(Target, AppendMe, AppendMeLen, 0);
454 void tmplput_ROOM_ACL(StrBuf *Target, WCTemplputParams *TP)
456 folder *Folder = (folder *)(TP->Context);
458 StrBufAppendPrintf(Target, "%ld", Folder->RAFlags, 0);
462 void tmplput_ROOM_QRFLAGS(StrBuf *Target, WCTemplputParams *TP)
464 folder *Folder = (folder *)(TP->Context);
465 StrBufAppendPrintf(Target, "%d", Folder->QRFlags);
470 void tmplput_ROOM_FLOORID(StrBuf *Target, WCTemplputParams *TP)
472 folder *Folder = (folder *)(TP->Context);
473 StrBufAppendPrintf(Target, "%d", Folder->floorid);
476 void tmplput_ROOM_LISTORDER(StrBuf *Target, WCTemplputParams *TP)
478 folder *Folder = (folder *)(TP->Context);
479 StrBufAppendPrintf(Target, "%d", Folder->listorder);
481 void tmplput_ROOM_VIEW(StrBuf *Target, WCTemplputParams *TP)
483 folder *Folder = (folder *)(TP->Context);
484 StrBufAppendPrintf(Target, "%d", Folder->view);
486 void tmplput_ROOM_DEFVIEW(StrBuf *Target, WCTemplputParams *TP)
488 folder *Folder = (folder *)(TP->Context);
489 StrBufAppendPrintf(Target, "%d", Folder->defview);
491 void tmplput_ROOM_LASTCHANGE(StrBuf *Target, WCTemplputParams *TP)
493 folder *Folder = (folder *)(TP->Context);
494 StrBufAppendPrintf(Target, "%d", Folder->lastchange);
496 void tmplput_ROOM_FLOOR_ID(StrBuf *Target, WCTemplputParams *TP)
498 folder *Folder = (folder *)(TP->Context);
499 const floor *Floor = Folder->Floor;
504 StrBufAppendPrintf(Target, "%d", Floor->ID);
507 void tmplput_ROOM_FLOOR_NAME(StrBuf *Target, WCTemplputParams *TP)
509 folder *Folder = (folder *)(TP->Context);
510 const floor *Floor = Folder->Floor;
515 StrBufAppendTemplate(Target, TP, Floor->Name, 0);
518 void tmplput_ROOM_FLOOR_NROOMS(StrBuf *Target, WCTemplputParams *TP)
520 folder *Folder = (folder *)(TP->Context);
521 const floor *Floor = Folder->Floor;
525 StrBufAppendPrintf(Target, "%d", Floor->NRooms);
530 int ConditionalRoomHas_UA_KNOWN(StrBuf *Target, WCTemplputParams *TP)
532 folder *Folder = (folder *)(TP->Context);
533 return (Folder->RAFlags & UA_KNOWN) != 0;
536 int ConditionalRoomHas_UA_GOTOALLOWED(StrBuf *Target, WCTemplputParams *TP)
538 folder *Folder = (folder *)(TP->Context);
539 return (Folder->RAFlags & UA_GOTOALLOWED) != 0;
542 int ConditionalRoomHas_UA_HASNEWMSGS(StrBuf *Target, WCTemplputParams *TP)
544 folder *Folder = (folder *)(TP->Context);
545 return (Folder->RAFlags & UA_HASNEWMSGS) != 0;
548 int ConditionalRoomHas_UA_ZAPPED(StrBuf *Target, WCTemplputParams *TP)
550 folder *Folder = (folder *)(TP->Context);
551 return (Folder->RAFlags & UA_ZAPPED) != 0;
555 void jsonRoomFlr(void)
557 /* Send as our own (application/json) content type */
558 hprintf("HTTP/1.1 200 OK\r\n");
559 hprintf("Content-type: application/json; charset=utf-8\r\n");
560 hprintf("Server: %s / %s\r\n", PACKAGE_STRING, ChrPtr(WC->serv_info->serv_software));
561 hprintf("Connection: close\r\n");
562 hprintf("Pragma: no-cache\r\nCache-Control: no-store\r\nExpires:-1\r\n");
564 DoTemplate(HKEY("json_roomflr"),NULL,&NoCtx);
574 WebcitAddUrlHandler(HKEY("json_roomflr"), jsonRoomFlr, 0);
577 RegisterNamespace("FLOOR:ID", 0, 0, tmplput_FLOOR_ID, CTX_FLOORS);
578 RegisterNamespace("FLOOR:NAME", 0, 1, tmplput_FLOOR_NAME, CTX_FLOORS);
579 RegisterNamespace("FLOOR:NROOMS", 0, 0, tmplput_FLOOR_NROOMS, CTX_FLOORS);
583 RegisterIterator("LKRA", 0, NULL, GetRoomListHashLKRA, NULL, DeleteHash, CTX_ROOMS, CTX_NONE, IT_FLAG_DETECT_GROUPCHANGE);
585 RegisterNamespace("ROOM:INFO:FLOORID", 0, 1, tmplput_ROOM_FLOORID, CTX_ROOMS);
586 RegisterNamespace("ROOM:INFO:NAME", 0, 1, tmplput_ROOM_NAME, CTX_ROOMS);
587 RegisterNamespace("ROOM:INFO:BASENAME", 0, 1, tmplput_ROOM_BASENAME, CTX_ROOMS);
588 RegisterNamespace("ROOM:INFO:LEVELNTIMES", 1, 2, tmplput_ROOM_LEVEL_N_TIMES, CTX_ROOMS);
590 RegisterNamespace("ROOM:INFO:ACL", 0, 1, tmplput_ROOM_ACL, CTX_ROOMS);
591 RegisterNamespace("ROOM:INFO:QRFLAGS", 0, 1, tmplput_ROOM_QRFLAGS, CTX_ROOMS);
592 RegisterNamespace("ROOM:INFO:LISTORDER", 0, 1, tmplput_ROOM_LISTORDER, CTX_ROOMS);
593 RegisterNamespace("ROOM:INFO:VIEW", 0, 1, tmplput_ROOM_VIEW, CTX_ROOMS);
594 RegisterNamespace("ROOM:INFO:DEFVIEW", 0, 1, tmplput_ROOM_DEFVIEW, CTX_ROOMS);
595 RegisterNamespace("ROOM:INFO:LASTCHANGE", 0, 1, tmplput_ROOM_LASTCHANGE, CTX_ROOMS);
596 RegisterNamespace("ROOM:INFO:FLOOR:ID", 0, 0, tmplput_ROOM_FLOOR_ID, CTX_ROOMS);
597 RegisterNamespace("ROOM:INFO:FLOOR:NAME", 0, 1, tmplput_ROOM_FLOOR_NAME, CTX_ROOMS);
598 RegisterNamespace("ROOM:INFO:FLOOR:NROOMS", 0, 0, tmplput_ROOM_FLOOR_NROOMS, CTX_ROOMS);
601 RegisterConditional(HKEY("COND:ROOM:FLAGS:UA_KNOWN"), 0, ConditionalRoomHas_UA_KNOWN, CTX_ROOMS);
602 RegisterConditional(HKEY("COND:ROOM:FLAGS:UA_GOTOALLOWED"), 0, ConditionalRoomHas_UA_GOTOALLOWED, CTX_ROOMS);
603 RegisterConditional(HKEY("COND:ROOM:FLAGS:UA_HASNEWMSGS"), 0, ConditionalRoomHas_UA_HASNEWMSGS, CTX_ROOMS);
604 RegisterConditional(HKEY("COND:ROOM:FLAGS:UA_ZAPPED"), 0, ConditionalRoomHas_UA_ZAPPED, CTX_ROOMS);
608 RegisterSortFunc(HKEY("byfloorroom"),
610 CompareRoomListByFloorRoomPrivFirst,
611 CompareRoomListByFloorRoomPrivFirstRev,
612 GroupchangeRoomListByFloorRoomPrivFirst,