Not logged in.  Login/Logout/Register | List snippets | | Create snippet | Upload image | Upload data

609
LINES

< > BotCompany Repo | #1004308 // Random Main v10

JavaX fragment (include)

1  
set flag Matches_fsi.
2  
3  
sbool assisting;
4  
5  
sclass RandomMain {
6  
  S generatorsID;
7  
  
8  
  JFrame frame;
9  
  JTable table, chatTable;
10  
  //JTextArea chat;
11  
  JTextField input;
12  
  L<S> log = synchroList();
13  
  L<S> recommendations;
14  
  L<Thread> thinkThreads = synchroList();
15  
  JLabel status;
16  
  S lastInput;
17  
  Map matchingSuggestion;
18  
  Map<S, ImmL<S>> allLogs = synchroMap();
19  
  JFrame analyzersFrame;
20  
  S dialogImageID;
21  
  JFrame dialogImageFrame;
22  
  
23  
  int listDelay = 2000;
24  
  int maxLineLength = 1000;
25  
  int maxLongStringLength = 100*1000;
26  
  
27  
  Bool thinking;
28  
  new Thinker thinker;
29  
  bool showCPU = true;
30  
  
31  
  S systemPrefix = "[system]";
32  
  
33  
  S dialog = "new";
34  
  
35  
  void randomMain() {
36  
    regularGC();
37  
    //substanceLAF("EmeraldDusk"); // Too dark!
38  
    //substanceLAF("ChallengerDeep"); // So purple!
39  
    //substance("MistAqua");
40  
    substance("Moderate");
41  
  
42  
    table = tableWithTooltips();
43  
    //chat = autoScroll(wordWrapTextArea());
44  
    chatTable = tableWithTooltips();
45  
    input = new JTextField;
46  
    status = jlabel("");
47  
    
48  
    S title = assisting ? "Assistance - " + autoFrameTitle() : autoFrameTitle();
49  
    frame = showFrame(title, vgrid(centerAndSouth(
50  
      //jtabs(1, "Chat", chat, "Details", chatTable),
51  
      chatTable,
52  
      input), centerAndSouth(table, jline(jMemoryInfo(), status))));
53  
    if (!assisting) hideConsole();
54  
    //setFrameIconLater(frame, "#1003593");
55  
    
56  
    addMenu(frame, "Random",
57  
      "Pop up last line in window (!pop)", r { pop(); },
58  
      "Delete last line (!delete)", r { post("!delete"); },
59  
      "Reload generators (!gen)", r { post("!gen"); },
60  
      "Restart app (!restart)", r { post("!restart"); },
61  
      "Restart Java engine (!fresh)", r { post("!fresh"); },
62  
      "Execute Java code (!j ...)", r { setInput("!j 1+2"); },
63  
      "Switch dialog (!dialog ...)", r { setInput("!dialog bla"); },
64  
      "Restore last input", r { if (nempty(lastInput)) setInput(lastInput); },
65  
      "Show raw dialog", r { showText("Raw Dialog", rawDialog()); },
66  
    );
67  
    
68  
    makeDialogsMenu();
69  
    
70  
    addMenu(frame, "Tools",
71  
      "Show Logic List", r { nohupJavax("#1004419"); }
72  
    );
73  
    
74  
    onEnter(input, r {
75  
      post();
76  
    });
77  
    
78  
    onDoubleClick(table, voidfunc(int row) {
79  
      chooseSuggestionForEditing(row);
80  
    });
81  
    
82  
    for (int i = 1; i <= 12; i++) {
83  
      final int _i = i;
84  
      registerFunctionKey(frame, i, r {
85  
        chooseSuggestionForEditing(_i-1);
86  
      });
87  
      registerShiftFunctionKey(frame, i, r {
88  
        chooseSuggestion(_i-1);
89  
      });
90  
      registerCtrlFunctionKey(frame, i, r {
91  
        // post user input and then choose suggestion for editing
92  
        Map<S, S> map = getTableLineAsMap(table, _i-1);
93  
        if (map == null) ret;
94  
        rankToTop(map);
95  
        S s = trim(unnull(map.get("Suggestion")));
96  
        logEvent("Suggestion chosen for editing with pre-post", mapPlus(map, "Row", _i, "Input", getInput(), "Top Suggesters", topSuggesters()));
97  
        post();
98  
        setInput(s);
99  
      });
100  
    }
101  
    
102  
    onDoubleClickOrEnter(chatTable, voidfunc(int row) {
103  
      Map map = getTableLineAsMap(chatTable, row);
104  
      if (map != null)
105  
        setInput(getString(map, "Text"));
106  
    });
107  
    
108  
    onUpdate(input, r { updateOnce(); });
109  
    
110  
    loadDialog();
111  
    logEvent("Starting");
112  
    
113  
    updateOnce();
114  
    
115  
    input.requestFocus();
116  
    
117  
    makeBot("Chat.", this);
118  
    
119  
    veryQuickJava_optimize();
120  
    
121  
    S last = last(log);
122  
    if (isAction(last) && confirmYesNo(input, "Run action? " + last(log)))
123  
      action(last);
124  
  }
125  
  
126  
  S getInput() {
127  
    ret joinLines(" # ", input.getText().trim());
128  
  }
129  
  
130  
  void post() {
131  
    postAsUser(getInput(), null);
132  
  }
133  
  
134  
  void postAsUser(S i, Map infos) {
135  
    if (inputAllowedByUser(i))
136  
      post(i, infos);
137  
  }
138  
  
139  
  void chooseSuggestionForEditing(int row) {
140  
    Map<S, S> map = getTableLineAsMap(table, row);
141  
    if (map == null) ret;
142  
    rankToTop(map);
143  
    S s = trim(unnull(map.get("Suggestion")));
144  
    logEvent("Suggestion chosen for editing", mapPlus(map, "Row", row+1, "Input", getInput(), "Top Suggesters", topSuggesters()));
145  
    setInput(s);
146  
  }
147  
  
148  
  void setInput(final S s) {
149  
    awtIfNecessary {
150  
      lastInput = input.getText();
151  
      input.setText(s);
152  
      input.selectAll();
153  
      input.requestFocus();
154  
    }
155  
  }
156  
  
157  
  void rankToTop(Map map) {
158  
    rankToTop(map, false);
159  
  }
160  
  
161  
  void rankToTop(Map map, bool removeISuggesters) {
162  
    if (map == null) ret;
163  
    O s = map.get("Suggesters");
164  
    // Table cells have been structure'd by dataToTable
165  
    L<S> sugg = s instanceof L ? (L) s : (L) unstructure((S) s);
166  
    
167  
    // These are cheaters!
168  
    if (removeISuggesters)
169  
      sugg = rejectWhere(func(S s) { s.endsWith("/i") }, sugg);
170  
    
171  
    thinker.rankToTop(first(sugg));
172  
  }
173  
  
174  
  void chooseSuggestion(int row) {
175  
    Map<S, S> map = getTableLineAsMap(table, row);
176  
    if (map == null) ret;
177  
    rankToTop(map);
178  
    S s = trim(unnull(map.get("Suggestion")));
179  
    if (empty(s)) ret;
180  
    
181  
    //logEvent("Suggestion chosen", mapPlus(map, "Row", row+1, "Input", getInput(), "Top Suggesters", topSuggesters));
182  
    setInput(s);
183  
    postAsUser(s, mapPlus(map, "Index", row+1));
184  
  }
185  
  
186  
  L topSuggesters() {
187  
    int n = 20;
188  
    n = min(n, tableRows(table));
189  
    new L topSuggesters;
190  
    for (int i = 0; i < n; i++)
191  
      topSuggesters.add(getTableLineAsMap(table, i));
192  
    //if (empty(topSuggesters)) topSuggesters = null;
193  
    ret topSuggesters;
194  
  }
195  
  
196  
  void logEvent(S type) {
197  
    logEvent(type, litmap());
198  
  }
199  
  
200  
  void logEvent(S type, Map map) {
201  
    logStructure(new File(dialogDir(), "event.log"),
202  
      ll(type, chatTime(), map));
203  
  }
204  
  
205  
  bool inputAllowedByUser(S i) {
206  
    ret !swic(i, systemPrefix);
207  
  }
208  
  
209  
  // may be called from other thread
210  
  void postSystemMessage(final S msg) {
211  
    if (empty(msg)) ret;
212  
    awtIfNecessary {
213  
      post(systemPrefix + " " + msg, litmap("By", "System"));
214  
    }
215  
  }
216  
  
217  
  S post(S i) {
218  
    ret post(i, null);
219  
  }
220  
  
221  
  S post(S i, Map infos) {
222  
    S chatTime = null;
223  
    try {
224  
      i = trim(i);
225  
      if (empty(i)) null;
226  
      //i = escapeNewLines(i);
227  
      
228  
      if (infos == null) {
229  
        infos = matchingSuggestion;
230  
        if (infos != null)
231  
          print("Ranking to top: " + struct(infos));
232  
        rankToTop(infos, true);
233  
      }
234  
      infos = mapPlus(infos, "Top Suggesters", topSuggesters());
235  
      
236  
      bool tooLong = l(i) > maxLongStringLength;
237  
      
238  
      if (l(i) > maxLineLength) {
239  
        S id = saveLongString(i);
240  
        i = substring(i, 0, maxLineLength) + "... [" + (tooLong ? "too " : "") + "long text " + id + "]";
241  
      }
242  
    } catch e {
243  
      printStackTrace(e);
244  
      i = systemPrefix + " " + exceptionToStringShort(e);
245  
    }
246  
    
247  
    S s = i + "\n";
248  
    //chat.append(escapeNewLines(i) + "\n");
249  
    chatTime = chatTime();
250  
    appendToFile(logFile(), "[" + chatTime + "] " + s);
251  
    logEvent("Posting", litmap("Text", i, "Infos", infos));
252  
    log.add(i);
253  
    updateChatTable();
254  
    input.selectAll();
255  
    updateOnce();
256  
    try {
257  
      action(i);
258  
    } catch e {
259  
      printStackTrace(e);
260  
      postSystemMessage(exceptionToStringShort(e));
261  
    }
262  
    ret chatTime;
263  
  }
264  
  
265  
  S dropActionPrefix(S s) {
266  
    if (s == null) null;
267  
    s = dropBracketPrefix(s); // e.g. "[bot]"
268  
    if (!s.startsWith("!")) null;
269  
    ret s.substring(1);
270  
  }
271  
  
272  
  bool isAction(S s) {
273  
    ret dropActionPrefix(s) != null;
274  
  }
275  
  
276  
  void action(S s) {
277  
    s = dropActionPrefix(s);
278  
    if (s == null) ret;
279  
  
280  
    final S _s = s;
281  
    thread "Action" {
282  
      JWindow _loading_window = showLoadingAnimation();
283  
      try {
284  
        genLog_set(getLog()); // 'case user needs it
285  
        gOtherLogs_set(getOtherLogs()); // ditto
286  
        randomsOwnCmds(_s);
287  
        systemCommands(_s, RandomMain.this);
288  
      } catch e {
289  
        printStackTrace(e);
290  
        postSystemMessage("Error - " + exceptionToStringShort(e));
291  
      } finally {
292  
        genLog_clear();
293  
        gOtherLogs_clear();
294  
        disposeWindow(_loading_window);
295  
      }
296  
    }
297  
  }
298  
  
299  
  volatile bool again;
300  
  
301  
  // This logic is bad...
302  
  void fillList(bool force) {
303  
    bool t = force || shouldUpdateList();
304  
    if (neq(t, thinking)) {
305  
      thinking = t;
306  
      setFrameIcon(table, t ? "#1003603" : "#1003593");
307  
    }
308  
    
309  
    if (!t) {
310  
      if (!force)
311  
        againl8r();
312  
    } else {
313  
      if (nempty(thinkThreads)) { again = true; ret; }
314  
      fillListImpl();
315  
    }
316  
  }
317  
  
318  
  void addKeys(L<Map> data) {
319  
    for (int i = 0; i < l(data); i++) {
320  
      int k = i+1;
321  
      S key = k <= 12 ? "F" + k : null;
322  
      Map m = litorderedmap("Key", key);
323  
      m.putAll(data.get(i));
324  
      data.set(i, m);
325  
    }
326  
  }
327  
  
328  
  // data should be in the un-stringified form
329  
  void removeMatchingLine(L<Map> data) {
330  
    S input = getInput();
331  
    matchingSuggestion = null;
332  
    for (int i = 0; i < l(data); i++)
333  
      if (eq(input, data.get(i).get("Suggestion"))) {
334  
        Map m = data.get(i);
335  
        matchingSuggestion = mapPlus(m, "Index", i+1);
336  
        pcall {
337  
          L suggesters = (L) m.get("Suggesters");
338  
          if (allEndWith(suggesters, "/i")) { // << These cheat too much
339  
            //print("Dropping matching suggestion from: " + struct(suggesters));
340  
            matchingSuggestion = null;
341  
          }
342  
        }
343  
        data.remove(i);
344  
        ret;
345  
      }
346  
  }
347  
  
348  
  void fillListImpl() {
349  
    thread "Fill List" {
350  
      try {
351  
        thinkThreads.add(currentThread());
352  
        final new L<Map> data;
353  
        
354  
        thinker.makeListData(cloneList(log), getInput(), getOtherLogs(), data);
355  
        
356  
        awt {
357  
          pcall {
358  
            removeMatchingLine(data);
359  
            addKeys(data);
360  
            dataToTable_uneditable(table, data);
361  
            tableColumnMaxWidth(table, 0, 30); // "Key" column
362  
          }
363  
          againl8r();
364  
        }
365  
      } finally {
366  
        thinkThreads.remove(currentThread());
367  
      if (again) { again = false; fillListImpl(); }
368  
      }
369  
    }
370  
  }
371  
  
372  
  void updateOnce() { fillList(true); }
373  
  
374  
  void againl8r() {
375  
    swingAfter(table, listDelay, r { fillList(false); });
376  
  }
377  
  
378  
  bool shouldUpdateList() {
379  
    bool result = false;
380  
    S text = " ";
381  
    if (getFrame(table).isFocused()) {
382  
      result = !mouseInComponent(table);
383  
      text = result ? " Thinking..." + (showCPU /*&& thinker.load != 0*/ ? " (" + thinker.load + "% CPU)" : "")
384  
        : "Not thinking cause you got the mouse in there";
385  
    }
386  
    status.setText(text);
387  
    ret result;
388  
  }
389  
  
390  
  // also called from outside
391  
  L<S> loadLog() {
392  
    log.clear();
393  
    log.addAll(scanEventLogForText(dialogDir()));
394  
    ret log;
395  
  }
396  
  
397  
  void loadAllLogs() {
398  
    allLogs.clear();
399  
    for (File f : findAllFiles(getProgramDir()))
400  
      if (f.getName().equals("event.log"))
401  
        allLogs.put(f.getParentFile().getName(), new ImmL(scanEventLogForText(f)));
402  
  }
403  
  
404  
  synchronized L<S> getLastFromLog(int n) {
405  
    ret cloneList(getLast(log, n));
406  
  }
407  
  
408  
  synchronized L<S> getLog() {
409  
    ret cloneList(log);
410  
  }
411  
  
412  
  File dialogDir() {
413  
    ret prepareProgramFile(dialog);
414  
  }
415  
  
416  
  File logFile() {
417  
    ret new File(dialogDir(), "log.txt");
418  
  }
419  
  
420  
  void switchDialog(final S name) {
421  
    swingAndWait(r {
422  
      dialog = name;
423  
      touchFile(new File(dialogDir(), "event.log"));
424  
      loadDialog();
425  
      loadAllLogs();
426  
      makeDialogsMenu();
427  
      setFrameTitle(dialog + " - " + getProgramTitle());
428  
  
429  
      // show special stuff for dialog, like an image
430  
      dialogImageID = trim(readTextFile(new File(dialogDir(), "image-id.txt")));
431  
      showDialogImage();
432  
    });
433  
  }
434  
  
435  
  void loadDialog() {
436  
    loadLog();
437  
    thinker = new Thinker;
438  
    thinker.startUp(this, log);
439  
    //chat.setText(joinLines(log));
440  
    updateChatTable();
441  
  }
442  
  
443  
  void randomsOwnCmds(S s) {
444  
    new Matches m;
445  
    
446  
    if "dialog *" {
447  
      switchDialog(m.unq(0));
448  
      // TODO: show current dialog somewhere else
449  
      //postSystemMessage("OK, dialog switched to " + quote(dialog));
450  
    }
451  
    
452  
    if "gen"
453  
      generators = null;
454  
    
455  
    if "delete"
456  
      updateChatTable();
457  
      
458  
    if "analyzers"
459  
      showAnalyzers();
460  
      
461  
    if (matchOneOf(s, m, "img *", "image *")) {
462  
      S imageID = m.fsi(0);
463  
      saveTextFile(new File(dialogDir(), "image-id.txt"), imageID);
464  
      dialogImageID = imageID;
465  
      showDialogImage();
466  
    }
467  
    
468  
    if (matchOneOf(s, m, "img", "image")) {
469  
      if (empty(dialogImageID))
470  
        postSystemMessage("No image set for this dialog");
471  
      else
472  
        postSystemMessage("Image for this dialog: " + fsI(dialogImageID));
473  
    }
474  
  }
475  
  
476  
  void showAnalyzers() {
477  
    if (analyzersFrame == null)
478  
      analyzersFrame = showFrame("Analyzers");
479  
  }
480  
  
481  
  void updateChatTable() {
482  
    awtIfNecessary {
483  
      new L data;
484  
      L<L> l = scanLog_safeUnstructure(new File(dialogDir(), "event.log"));
485  
      for (int i = 0; i < l(l); i++) pcall {
486  
        L a = l.get(i), prev = get(l, i-1);
487  
        if (firstIs(a, "Posting")) {
488  
          Map map = cast get(a, 2);
489  
          S text = getString(map, "Text").trim();
490  
          if (eq(text, "!delete")) { removeLast(data); continue; }
491  
          S idx = "";
492  
          Map infos = cast map.get("Infos");
493  
          if (infos != null && infos.containsKey("Index"))
494  
            idx = str(infos.get("Index"));
495  
          else pcall {
496  
            //printStruct("prev: ", prev);
497  
            if (prev != null && firstIs(prev, "Suggestion chosen for editing")) {
498  
              Map m = getMap(prev, 2);
499  
              S suggestion = getString(m, "Suggestion");
500  
              //print("Suggestion: " + structure(suggestion));
501  
              idx = str(get(m, "Row"));
502  
              if (neq(suggestion, text))
503  
                idx += "?";
504  
            }
505  
          }
506  
          data.add(litorderedmap(
507  
            "#", "", // not known yet cause of !delete statements
508  
            "Text", escapeNewLines(text),
509  
            "Sugg." /* Suggestion Index */, idx));
510  
        }
511  
      }
512  
      addNumbering(data);
513  
      dataToTable_uneditable(chatTable, data);
514  
      tableColumnMaxWidth(chatTable, 0, 30); // enough for 3 digits?
515  
      tableColumnMaxWidth(chatTable, 2, 40); // enough for 2 digits and a "?"?
516  
      scrollTableDownIn(chatTable, 50);
517  
    }
518  
  }
519  
  
520  
  void addNumbering(L<Map> data) {
521  
    int n = 0;
522  
    for (int i = l(data)-1; i >= 0; i--)
523  
      data.get(i).put("#", ++n);
524  
  }
525  
  
526  
  synchronized S saveLongString(S s) {
527  
    s = substring(s, 0, maxLongStringLength);
528  
    S id;
529  
    File f;
530  
    do {
531  
      id = randomID(10);
532  
      f = getProgramFile("long-strings/" + id);
533  
    } while (f.exists());
534  
    
535  
    saveTextFile(f, s);
536  
    ret id;
537  
  }
538  
  
539  
  O generators;
540  
  
541  
  void makeGenerators(L<Gen> l) {
542  
    synchronized(main.class) {
543  
      if (!isSnippetID(generatorsID)) fail("No generators ID set");
544  
      if (generators == null)
545  
        generators = hotwire(generatorsID);
546  
    }
547  
    new L l2;
548  
    callOpt(generators, "makeGenerators", l2);
549  
    callOpt(generators, "deterministicGenerators", l2);
550  
    l.addAll((L) quickImport(l2));
551  
  }
552  
  
553  
  S rawDialog() {
554  
    ret fromLines(log);
555  
  }
556  
  
557  
  void makeDialogsMenu() {
558  
    new L items;
559  
    for (File dir : listDirs(getProgramDir()))
560  
      if (containsFile(dir, "event.log")) {
561  
        fS dialog = dir.getName();
562  
        items.add(dialog);
563  
        items.add(r { switchDialog(dialog); });
564  
      }
565  
      
566  
    addMenu(frame, "Dialogs", items);
567  
  }
568  
  
569  
  L<ImmL<S>> getOtherLogs() {
570  
    // TODO: sort by latest or smth?
571  
    ret valuesList(mapMinus(allLogs, dialog));
572  
  }      
573  
  
574  
  void hideDialogImage() {
575  
    if (dialogImageFrame != null) {
576  
      disposeFrame(dialogImageFrame);
577  
      dialogImageFrame = null;
578  
    }
579  
  }
580  
  
581  
  void showDialogImage() {
582  
    hideDialogImage();
583  
    if (nempty(dialogImageID)) {
584  
    S title = getSnippetTitle(dialogImageID) + " [" + fsI(dialogImageID) + "/" + dialog + "]";
585  
      dialogImageFrame = getFrame(showImage(dialogImageID, title));
586  
    }
587  
  }
588  
  
589  
  S answer(S s) {
590  
    print("answer: " + s);
591  
    new Matches m;
592  
    if "please post * to chat file *" {
593  
      final S line = m.unq(0);
594  
      File file = new File(m.unq(1));
595  
      File myFile = new File(dialogDir(), "event.log");
596  
      print("Comparing files: " + file + " / " + myFile);
597  
      if (sameFile(file, myFile)) {
598  
        print("Foreign-posting: " + quote(line));
599  
        ret ok(swingAndWait(func { post(line) }));
600  
      }
601  
    }
602  
    null;
603  
  }
604  
  
605  
  void pop() {
606  
    S text = last(log);
607  
    if (nempty(text)) showText(text);
608  
  }
609  
}

Author comment

Began life as a copy of #1004031

download  show line numbers  debug dex  old transpilations   

Travelled to 16 computer(s): aoiabmzegqzx, bhatertpkbcr, cbybwowwnfue, cfunsshuasjs, ddnzoavkxhuk, gwrvuhgaqvyk, ishqpsrjomds, lpdgvwnxivlt, mqqgnosmbjvj, onxytkatvevr, pyentgdyhuwx, pzhvpgtvlbxg, tslmcundralx, tvejysmllsmz, vouqrxazstgt, xrpafgyirdlv

No comments. add comment

Snippet ID: #1004308
Snippet name: Random Main v10
Eternal ID of this version: #1004308/2
Text MD5: 706da8b26d71ac5b611cf1bf27251849
Author: stefan
Category: javax / talking robots
Type: JavaX fragment (include)
Public (visible to everyone): Yes
Archived (hidden from active list): No
Created/modified: 2020-07-02 11:52:57
Source code size: 16821 bytes / 609 lines
Pitched / IR pitched: No / No
Views / Downloads: 1019 / 1541
Version history: 1 change(s)
Referenced in: [show references]