QB64.org Forum

Active Forums => Programs => Topic started by: bplus on January 13, 2022, 11:46:33 am

Title: Boggle play against AI - WIP
Post by: bplus on January 13, 2022, 11:46:33 am
https://en.wikipedia.org/wiki/Boggle

The AI is going to be handicapped starting with only 1 point words (3 or 4 letters words from Scrabble Dictionary that don't use Q (that is 2 points)) and maybe a shorter time limit too.

Just got started last night from trying to figure out Dimster's "Babble" Game, I thought he might mean Boggle?

Any way got a board working:
Code: QB64: [Select]
  1. _Title "Boggle 1" ' b+ start 2022-01-12
  2. ' Scabble Word List and Dictionary
  3. ' ref dictionary: https://boardgames.stackexchange.com/questions/38366/latest-collins-scrabble-words-list-in-text-file
  4. ' Die configurations
  5. ' https://boardgames.stackexchange.com/questions/29264/boggle-what-is-the-dice-configuration-for-boggle-in-various-languages
  6. ' Thank you!
  7.  
  8. Dim Shared Board$(3, 3)
  9. Dim Shared As Long f48, f30, dx(7), dy(7)
  10. Screen _NewImage(800, 600, 32)
  11. _ScreenMove 200, 100
  12.     Cls
  13.     NewBoard
  14.     ' display timer and allow input of words from user for 3 minutes
  15.     ' meanwhile AI will calc all the 1 point words it can from board
  16.  
  17.     Sleep
  18.  
  19. Sub NewBoard
  20.     Static BeenHere, Di$(), Numbers()
  21.     Dim As Long i, r, c, row, col
  22.     If BeenHere = 0 Then 'load and initialize
  23.         f48 = _LoadFont("Arial.ttf", 48, "MONOSPACE")
  24.         f30 = _LoadFont("Arial.ttf", 30, "MONOSPACE")
  25.         If f48 <= 0 Then Print "Sub NewBoard: Font did not load, goodbye.": End
  26.         dx(0) = -1: dy(0) = -1 ' this is for AI to find words
  27.         dx(1) = 0: dy(1) = -1
  28.         dx(2) = 1: dy(2) = -1
  29.         dx(3) = -1: dy(3) = 0
  30.         dx(4) = 1: dy(4) = 0
  31.         dx(5) = -1: dy(5) = 1
  32.         dx(6) = 0: dy(6) = 1
  33.         dx(7) = 1: dy(7) = 1
  34.         Dim Di$(0 To 15) ' this for 16 di, 6 letters per
  35.         Di$(1) = "RIFOBX"
  36.         Di$(2) = "IFEHEY"
  37.         Di$(3) = "DENOWS"
  38.         Di$(4) = "UTOKND"
  39.         Di$(5) = "HMSRAO"
  40.         Di$(6) = "LUPETS"
  41.         Di$(7) = "ACITOA"
  42.         Di$(8) = "YLGKUE"
  43.         Di$(9) = "QBMJOA"
  44.         Di$(10) = "EHISPN"
  45.         Di$(11) = "VETIGN"
  46.         Di$(12) = "BALIYT"
  47.         Di$(13) = "EZAVND"
  48.         Di$(14) = "RALESC"
  49.         Di$(15) = "UWILRG"
  50.         Di$(0) = "PACEMD"
  51.         Dim Numbers(0 To 15) ' for shuffling die order
  52.         For i = 0 To 15
  53.             Numbers(i) = i
  54.         Next
  55.         BeenHere = -1
  56.     End If
  57.     For i = 15 To 1 Step -1 'shuffle die
  58.         Swap Numbers(i), Numbers(Int(Rnd * (i + 1)))
  59.     Next
  60.     'For i = 1 To 16: Print Numbers(i);: Next: Print   ' check the shuffle
  61.     For i = 0 To 15 'choosing random face of die = 1 Letter
  62.         Index2ColRow i, c, r
  63.         Board$(c, r) = Mid$(Di$(Numbers(i)), Int(Rnd * 6) + 1, 1)
  64.     Next
  65.     _Font f48
  66.     For row = 0 To 3 '  display the board
  67.         For col = 0 To 3
  68.             Line ((col + 1) * 60 - 5, (row + 1) * 60 - 5)-Step(54, 54), &HFF2020FF, BF 'face color or die
  69.             If Board$(col, row) = "Q" Then 'If face has a Q it is supposed to be "Qu"
  70.                 _Font f30
  71.                 Color &HFF661111 'shade
  72.                 _PrintString ((col + 1) * 60 - 4, (row + 1) * 60 + 11), "Q"
  73.                 _PrintString ((col + 1) * 60 + 24, (row + 1) * 60 + 11), "U"
  74.                 Color &HFFBBBBBB 'letter
  75.                 _PrintString ((col + 1) * 60 - 7, (row + 1) * 60 + 9), "Q"
  76.                 _PrintString ((col + 1) * 60 + 22, (row + 1) * 60 + 9), "U"
  77.                 _Font f48
  78.             Else
  79.                 Color &HFF661111 'shade
  80.                 _PrintString ((col + 1) * 60 + 2, (row + 1) * 60 + 2), Board$(col, row)
  81.                 Color &HFFBBBBBB 'letter
  82.                 _PrintString ((col + 1) * 60, (row + 1) * 60), Board$(col, row)
  83.             End If
  84.         Next
  85.     Next
  86.     _Font 16
  87.  
  88. Function ColRow2Index& (row As Long, col As Long) ' convert a board letter to index (not needed yet?)
  89.     ColRow2Index& = row * 4 + col
  90. Sub Index2ColRow (indexIn As Long, rowOut As Long, colOut As Long) 'convert die index to board col, row
  91.     colOut = indexIn Mod 4: rowOut = indexIn \ 4
  92.  
Title: Re: Boggle play against AI - WIP
Post by: bplus on January 13, 2022, 06:14:12 pm
Dang the Collins Dictionary is too big to load, QB64 keeps bugging out, no error message, just quits.
279,496 words
Title: Re: Boggle play against AI - WIP
Post by: bplus on January 14, 2022, 08:15:13 am
The amount of words I could store and not in terrible time was about 260,000 before QB64 (on my Windows 10 laptop system) bugged out without so much as error message, at least it didn't crash the whole system. Tried Open For Input and Open For Binary, one gulp method. OK it's 2.96 MB according to properties Window.

I have already worked a fairly practical workaround by limiting the words by number of letters, another maybe not to load the dictionary at all and just search for words in the hard disk by turning it into a Random access file or something.

But I wonder if there is another way to get the dictionary loaded into program whole. Is about 1 MB the limit for variable length string memory?
Title: Re: Boggle play against AI - WIP
Post by: bplus on January 14, 2022, 08:42:07 am
Oh boy! Just got the trickiest Function working :)

It's the one that confirms a word can or can not be legally built from a given board. Interestingly it uses a recursive function to seek out the rest of the word from a given letter location once the first letter has been matched from the game board. I am a bit shocked, it's like magic :)

 (Of course the shock would have been worse had it worked right off.)

Update: OK same logic bug happened in the recursive helper function, more subtle, I think I have them exterminated now.
Title: Re: Boggle play against AI - WIP
Post by: bplus on January 14, 2022, 12:37:46 pm
That AI looks like it will need more handicap, here is what it came up with in 30 secs. Looks like that was enough time to test the whole dang OnePointList$(). I've still got to work out dealing with special case Q amongst checking player redundant words, scoring and handicapping.

The first test of AI play:
  [ This attachment cannot be displayed inline in 'Print Page' view ]  

WIP:
Code: QB64: [Select]
  1. _Title "Boggle 1" ' b+ start 2022-01-12
  2. ' Scabble Word List and Dictionary
  3. ' ref dictionary: https://boardgames.stackexchange.com/questions/38366/latest-collins-scrabble-words-list-in-text-file
  4. ' Die configurations
  5. ' https://boardgames.stackexchange.com/questions/29264/boggle-what-is-the-dice-configuration-for-boggle-in-various-languages
  6. ' Thank you!
  7.  
  8. ' 2022-01-14 status have the board display separated out from New Board and Game initialization
  9. ' have timer system and display working next to board display
  10. ' have player input system working
  11. '   1. check word in dictionary working
  12. '   2. check that word can legally be built from board is funky! sometimes works sometimes not 50/50 for legit builds yuck
  13. '     How to debug? Todays task is get this abolutely critical function working so the AI would be fairly easy to do.
  14. ' Oh I had to modify the Collins Word list to get a size that would fit into a variable length string array, 279496 words
  15. ' kept crashing QB64 without error messages when reached about 260,000 words so now I have a list for words with lengths of
  16. ' between 3 and 10, still 199,651 words!
  17. ' 2202-01-14 Hurray! Function wordBuildOK& is fixed, a very simple logic bug. I continued to check for builds of word after
  18. ' I got confirmation one of the builds was possible. It kept checking if there was more than one place the first letter of word
  19. ' appeared. The function is crucial for AI testing if words are buildable. I can get started on that today now that bug is
  20. ' fixed. Man! I had this fixed almost immediately, I had to exit the Function when the recursive function findCell& called from
  21. ' it found a positive result. AI is working.
  22.  
  23. Const TimeLimit = 180 ' actual game time is 3 minutes = 180 secs
  24. Dim Shared Board$(3, 3), WordList$(1 To 199651), OnePointList$(1 To 6955)
  25. Dim Shared As Long f48, f30, dx(7), dy(7)
  26. Dim Shared As Double BoggleTime, elapsed
  27. Dim w$, k$, player$, AI$
  28. Dim As Long printLine
  29. Screen _NewImage(800, 600, 32)
  30. _ScreenMove 200, 100
  31.     NewBoard
  32.     DisplayBoard
  33.     elapsed = 0
  34.     'Do ' display timer and allow input of words from user for 3 minutes
  35.     printLine = 20
  36.     player$ = "": w$ = ""
  37.     While elapsed < TimeLimit ' do stuff   <<<<<<<<<<<<<<<<<<<<<< off while debug wordBuildOK
  38.         Cls
  39.         DisplayBoard
  40.         elapsed = Timer(.01) - BoggleTime
  41.         If elapsed < 0 Then elapsed = 24 * 60 * 60 + Timer(.01) - BoggleTime ' midnight problem add aday of seconds to timer and subtr boogle
  42.         _Font f48
  43.         Line (300, 240)-Step(180, 60), &HFF000000, BF ' blackout last time
  44.         Color &HFFFFFF00
  45.         _PrintString (300, 240), _Trim$(Str$(TimeLimit - Int(elapsed)))
  46.  
  47.         _Font 16
  48.         Color _RGB32(200, 200, 255)
  49.         k$ = UCase$(InKey$)
  50.         If Len(k$) Then 'handle 1 and 2 char key presses, maybe replace with _keyhit later
  51.             Select Case Len(k$)
  52.                 Case 1
  53.                     Select Case Asc(k$)
  54.                         Case 3 'Ctrl + C   another way to clear?
  55.                             w$ = ""
  56.                         Case 8 ' backspace          more to do
  57.                             If Len(w$) Then w$ = Left$(w$, Len(w$) - 1)
  58.                         Case 13
  59.                             If wordBuildOK&(w$) Then
  60.                                 If Find&(WordList$(), w$) Then ' check words before add to player$
  61.                                     If player$ = "" Then player$ = w$ Else player$ = player$ + " " + w$
  62.                                 End If
  63.                             End If
  64.                             w$ = ""
  65.                         Case 27 'esc
  66.                             w$ = "" ': exit ?
  67.                         Case Else
  68.                             If 63 < Asc(k$) And Asc(k$) < 91 Then w$ = w$ + k$
  69.                     End Select
  70.             End Select
  71.         End If
  72.         Locate printLine, 1: Print w$
  73.         Locate printLine + 2, 1: Print player$
  74.  
  75.         _Display
  76.         _Limit 60
  77.     Wend
  78.     _PrintString (400, 240), "Times up!"
  79.  
  80.     'lets see what the AI comes up with
  81.     AI$ = AIwords$(30) ' try 30 secs for starters
  82.     Locate printLine + 6, 1
  83.     Print "AI: "; AI$
  84.  
  85.     Sleep
  86.  
  87. Sub NewBoard
  88.     Static BeenHere, Di$(), Numbers()
  89.     Dim As Long i, r, c
  90.  
  91.     If BeenHere = 0 Then 'load and initialize all the one time stuff
  92.         ' load fonts
  93.         f48 = _LoadFont("Arial.ttf", 48, "MONOSPACE")
  94.         f30 = _LoadFont("Arial.ttf", 30, "MONOSPACE")
  95.         If f48 <= 0 Then Print "Sub NewBoard: Font did not load, goodbye.": End
  96.  
  97.         'load abrev Dictionary ======================================== comment out while debug wordBuildOK
  98.         Open "3 to 10 Letter Words.txt" For Input As #1
  99.         Print "Loading Dictionary..."
  100.         While Not EOF(1)
  101.             i = i + 1
  102.             Input #1, WordList$(i)
  103.             'Cls: Locate 2, 1: Print i
  104.         Wend
  105.         Close #1
  106.  
  107.         ' test load of file, find last 10 items
  108.         'For i = 199651 - 10 To 199651
  109.         '    Print WordList$(i)
  110.         'Next
  111.         'Sleep    OK loading
  112.  
  113.         Open "Boggle 1 Point Words.txt" For Input As #1
  114.         Print "Loading Boggle 1 Point Words.txt for AI..."
  115.         i = 0
  116.         While Not EOF(1)
  117.             i = i + 1
  118.             Input #1, OnePointList$(i)
  119.         Wend
  120.         Close #1
  121.  
  122.         ' load dx(), dy() for testing the legality of words built from board
  123.         dx(0) = -1: dy(0) = -1 ' this is for AI to find words
  124.         dx(1) = 0: dy(1) = -1
  125.         dx(2) = 1: dy(2) = -1
  126.         dx(3) = -1: dy(3) = 0
  127.         dx(4) = 1: dy(4) = 0
  128.         dx(5) = -1: dy(5) = 1
  129.         dx(6) = 0: dy(6) = 1
  130.         dx(7) = 1: dy(7) = 1
  131.  
  132.         ' These are the 16 Dice with 6 Faces of a Letter need for Boggle
  133.         Dim Di$(0 To 15) ' this for 16 di, 6 letters per
  134.         Di$(1) = "RIFOBX"
  135.         Di$(2) = "IFEHEY"
  136.         Di$(3) = "DENOWS"
  137.         Di$(4) = "UTOKND"
  138.         Di$(5) = "HMSRAO"
  139.         Di$(6) = "LUPETS"
  140.         Di$(7) = "ACITOA"
  141.         Di$(8) = "YLGKUE"
  142.         Di$(9) = "QBMJOA"
  143.         Di$(10) = "EHISPN"
  144.         Di$(11) = "VETIGN"
  145.         Di$(12) = "BALIYT"
  146.         Di$(13) = "EZAVND"
  147.         Di$(14) = "RALESC"
  148.         Di$(15) = "UWILRG"
  149.         Di$(0) = "PACEMD"
  150.  
  151.         Dim Numbers(0 To 15) ' load numbers for shuffling die order
  152.         For i = 0 To 15
  153.             Numbers(i) = i
  154.         Next
  155.         BeenHere = -1
  156.     End If
  157.  
  158.     'now get the game going
  159.     For i = 15 To 1 Step -1 'shuffle die
  160.         Swap Numbers(i), Numbers(Int(Rnd * (i + 1)))
  161.     Next
  162.     'For i = 1 To 16: Print Numbers(i);: Next: Print   ' check the shuffle
  163.     For i = 0 To 15 'choosing random face of die = 1 Letter
  164.         Index2ColRow i, c, r
  165.         Board$(c, r) = Mid$(Di$(Numbers(i)), Int(Rnd * 6) + 1, 1)
  166.     Next
  167.     ' now set timer + 180
  168.     BoggleTime = Timer(.01)
  169.     _Font 16
  170.  
  171. Sub DisplayBoard
  172.     Dim row, col
  173.     _Font f48
  174.     For row = 0 To 3 '  display the board
  175.         For col = 0 To 3
  176.             Line ((col + 1) * 60 - 5, (row + 1) * 60 - 5)-Step(54, 54), &HFF2020FF, BF 'face color or die
  177.             If Board$(col, row) = "Q" Then 'If face has a Q it is supposed to be "Qu"
  178.                 _Font f30
  179.                 Color &HFF661111 'shade
  180.                 _PrintString ((col + 1) * 60 - 4, (row + 1) * 60 + 11), "Q"
  181.                 _PrintString ((col + 1) * 60 + 24, (row + 1) * 60 + 11), "U"
  182.                 Color &HFFBBBBBB 'letter
  183.                 _PrintString ((col + 1) * 60 - 7, (row + 1) * 60 + 9), "Q"
  184.                 _PrintString ((col + 1) * 60 + 22, (row + 1) * 60 + 9), "U"
  185.                 _Font f48
  186.             Else
  187.                 Color &HFF661111 'shade
  188.                 _PrintString ((col + 1) * 60 + 2, (row + 1) * 60 + 2), Board$(col, row)
  189.                 Color &HFFBBBBBB 'letter
  190.                 _PrintString ((col + 1) * 60, (row + 1) * 60), Board$(col, row)
  191.             End If
  192.         Next
  193.     Next
  194.  
  195. Function ColRow2Index& (row As Long, col As Long) ' convert a board letter to index (not needed yet?)
  196.     ColRow2Index& = row * 4 + col
  197. Sub Index2ColRow (indexIn As Long, rowOut As Long, colOut As Long) 'convert die index to board col, row
  198.     colOut = indexIn Mod 4: rowOut = indexIn \ 4
  199.  
  200. Sub Split (SplitMeString As String, delim As String, loadMeArray() As String)
  201.     Dim curpos As Long, arrpos As Long, LD As Long, dpos As Long 'fix use the Lbound the array already has
  202.     curpos = 1: arrpos = LBound(loadMeArray): LD = Len(delim)
  203.     dpos = InStr(curpos, SplitMeString, delim)
  204.     Do Until dpos = 0
  205.         loadMeArray(arrpos) = Mid$(SplitMeString, curpos, dpos - curpos)
  206.         arrpos = arrpos + 1
  207.         If arrpos > UBound(loadMeArray) Then ReDim _Preserve loadMeArray(LBound(loadMeArray) To UBound(loadMeArray) + 1000) As String
  208.         curpos = dpos + LD
  209.         dpos = InStr(curpos, SplitMeString, delim)
  210.     Loop
  211.     loadMeArray(arrpos) = Mid$(SplitMeString, curpos)
  212.     ReDim _Preserve loadMeArray(LBound(loadMeArray) To arrpos) As String 'get the ubound correct
  213.  
  214. Function Find& (SortedArr$(), x$) ' if I am using this only to find words in dictionary, I can mod to optimize
  215.     Dim As Long low, hi, test
  216.     low = LBound(SortedArr$): hi = UBound(SortedArr$)
  217.     While low <= hi
  218.         test = Int((low + hi) / 2)
  219.         If SortedArr$(test) = x$ Then
  220.             Find& = test: Exit Function
  221.         Else
  222.             If SortedArr$(test) < x$ Then low = test + 1 Else hi = test - 1
  223.         End If
  224.     Wend
  225.  
  226. Function wordBuildOK& (w$) ' this function checks to see that the was constructed (or is constructable with the given board).
  227.     Dim As Long r, c, test
  228.     Dim copy$(-1 To 4, -1 To 4), first$
  229.     For r = 0 To 3
  230.         For c = 0 To 3
  231.             copy$(c, r) = Board$(c, r)
  232.         Next
  233.     Next
  234.  
  235.     first$ = Mid$(w$, 1, 1)
  236.     For r = 0 To 3
  237.         For c = 0 To 3
  238.             If copy$(c, r) = first$ Then 'cell letter matches first letter in word
  239.                 test = findCell&(c, r, w$, 2, copy$())
  240.                 If test Then wordBuildOK& = -1: Exit Function ' ah ha! maybe it keeps trying when we are supposed to be done, fix?
  241.             End If
  242.         Next
  243.     Next
  244.  
  245. 'recursively called starting from wordBuildOK&
  246. Function findCell& (startX As Long, startY As Long, word$, index As Long, Arr$()) ' want to setup recursive searcher
  247.     Dim As Long d, x, y, i, r, c, test
  248.     Dim w$
  249.     'make own set of variables for this function  (attempt to debug but did not fix anything)
  250.     Dim a$(-1 To 4, -1 To 4)
  251.     For r = 0 To 3
  252.         For c = 0 To 3
  253.             a$(c, r) = Arr$(c, r)
  254.         Next
  255.     Next
  256.     i = index: w$ = word$: y = startY: x = startX
  257.     If i > Len(w$) Then findCell = -1: Exit Function
  258.     a$(x, y) = "" 'so wont be used again
  259.     For d = 0 To 7
  260.         If a$(x + dx(d), y + dy(d)) = Mid$(w$, i, 1) Then
  261.             test = findCell&(x + dx(d), y + dy(d), w$, i + 1, a$())
  262.             If test Then findCell& = -1: Exit Function
  263.         End If
  264.     Next
  265.  
  266. Function AIwords$ (timeLimit As Long) 'returns a space delimiter string of 1 point words that can be constructed from board in limited time
  267.     Dim As Double startTime, checkTime
  268.     Dim As Long i, r, c, OK, dp, l, ub
  269.     Dim l$, letters$, b$
  270.     startTime = Timer(.01)
  271.     ub = UBound(OnePointList$)
  272.     ' get a non redundant list of letters from board
  273.     For r = 0 To 3
  274.         For c = 0 To 3
  275.             l$ = Board$(c, r)
  276.             If (r = 0) And (c = 0) Then
  277.                 letters$ = l$
  278.             Else
  279.                 If InStr(letters$, l$) <= 0 Then '  insrt letter
  280.                     OK = 0
  281.                     For i = 1 To Len(letters$)
  282.                         If Asc(l$) < Asc(letters$, i) Then ' insert spotted
  283.                             If i = 1 Then
  284.                                 letters$ = l$ + letters$: OK = -1: Exit For
  285.                             Else
  286.                                 letters$ = Mid$(letters$, 1, i - 1) + l$ + Mid$(letters$, i)
  287.                                 OK = -1: Exit For
  288.                             End If
  289.                         End If
  290.                     Next
  291.                     If OK = 0 Then letters$ = letters$ + l$
  292.                 End If
  293.             End If
  294.         Next
  295.     Next
  296.     'check if this is OK so far  OK finally!  This is 3rd time I needed to exit when found
  297.     ' AIwords$ = letters$
  298.     'now letters of board are in alpha order
  299.     dp = 1 'place in dict
  300.     For l = 1 To Len(letters$) ' advance place in list$ by one until the word > letter
  301.         While Asc(OnePointList$(dp), 1) < Asc(letters$, l)
  302.             dp = dp + 1
  303.             If dp > ub Then GoTo fini
  304.             If Timer(.01) - startTime < 0 Then checkTime = Timer(.01) + 24 * 60 * 60 Else checkTime = Timer(.01)
  305.             If checkTime - startTime > timeLimit Then GoTo fini
  306.         Wend
  307.         'now start testing words
  308.         While Asc(OnePointList$(dp), 1) = Asc(letters$, l)
  309.             If wordBuildOK&(OnePointList$(dp)) Then
  310.                 If b$ = "" Then b$ = OnePointList$(dp) Else b$ = b$ + " " + OnePointList$(dp)
  311.             End If
  312.             dp = dp + 1
  313.             If dp > ub Then GoTo fini
  314.             If Timer(.01) - startTime < 0 Then checkTime = Timer(.01) + 24 * 60 * 60 Else checkTime = Timer(.01)
  315.             If checkTime - startTime > timeLimit Then GoTo fini
  316.         Wend
  317.     Next
  318.  
  319.     fini:
  320.     AIwords$ = b$
Title: Re: Boggle play against AI - WIP
Post by: Statsman1 on January 14, 2022, 01:05:31 pm
@bplus - Dude, you are on FIRE.  Fantastic stuff so quickly!
Title: Re: Boggle play against AI - WIP
Post by: bplus on January 14, 2022, 03:15:20 pm
@bplus - Dude, you are on FIRE.  Fantastic stuff so quickly!

Gotta say, @Statsman1 you are helping fan the fire, thanks for your enthusiastic interest.
Title: Re: Boggle play against AI - WIP
Post by: Statsman1 on January 14, 2022, 03:39:49 pm
Gotta say, @Statsman1 you are helping fan the fire, thanks for your enthusiastic interest.

I love word games, so this is just great stuff.  I really appreciate that you would do all of this.
Title: Re: Boggle play against AI - WIP
Post by: bplus on January 14, 2022, 03:51:41 pm
I love word games, so this is just great stuff.  I really appreciate that you would do all of this.

Ah word games, checkout @Qwerkey plus crosswords and another about snaking word around a grid maybe like Boggle? It was from a newspaper puzzle and it was awhile ago and I may have author confused with someone else?

Bplus also did WordSearch had a pretty good package for Rosetta Code Challenge but interacting with Richard Frost and his work of getting all the elements of periodic table to fit in a grid really improved my game. I think I got to point of building word searches for your own list of items.
Title: Re: Boggle play against AI - WIP
Post by: bplus on January 15, 2022, 12:13:53 pm
OK I have basic game roughed out. Now I am handling the Q(u) Letter better still needs more testing. I need a qu? word that is 3 letters with u, to see if you type 2 letters the first a q and enter q? the word will be added to your list. BTW the 1 Point List that the AI uses doesn't have any Q words because originally I thought Q words were 2 points. No Q words contain the hidden letter U which could potentially make the word a letter longer that the one you type and points are awarded by number of letters. All the goofiness in coding could have been simplified by having 1 die with Q and maybe a couple extra u's and other vowels. That may be a mod down the line from here 17 dice, the 17th with aeiouu. This game is boring if have a whole bunch of constanents (sp? for not-a-vowel).

Now there is a Sleep between seeing your list and AI's before matching words are removed from both lists and final score calculated and shown. Yours will likely be 0 unless you come up with 5 or more letter words but at lest all your 3-4 letter word will negate the AI's.

Code: QB64: [Select]
  1. _Title "Boggle 1" ' b+ start 2022-01-12
  2. ' Scabble Word List and Dictionary
  3. ' ref dictionary: https://boardgames.stackexchange.com/questions/38366/latest-collins-scrabble-words-list-in-text-file
  4. ' Die configurations
  5. ' https://boardgames.stackexchange.com/questions/29264/boggle-what-is-the-dice-configuration-for-boggle-in-various-languages
  6. ' Thank you!
  7.  
  8. ' 2022-01-14 status have the board display separated out from New Board and Game initialization
  9. ' have timer system and display working next to board display
  10. ' have player input system working
  11. '   1. check word in dictionary working
  12. '   2. check that word can legally be built from board is funky! sometimes works sometimes not 50/50 for legit builds yuck
  13. '     How to debug? Todays task is get this abolutely critical function working so the AI would be fairly easy to do.
  14. ' Oh I had to modify the Collins Word list to get a size that would fit into a variable length string array, 279496 words
  15. ' kept crashing QB64 without error messages when reached about 260,000 words so now I have a list for words with lengths of
  16. ' between 3 and 10, still 199,651 words!
  17. ' 2202-01-14 Hurray! Function wordBuildOK& is fixed, a very simple logic bug. I continued to check for builds of word after
  18. ' I got confirmation one of the builds was possible. It kept checking if there was more than one place the first letter of word
  19. ' appeared. The function is crucial for AI testing if words are buildable. I can get started on that today now that bug is
  20. ' fixed. Man! I had this fixed almost immediately, I had to exit the Function when the recursive function findCell& called from
  21. ' it found a positive result.
  22.  
  23. ' 2022-01-15 qw$() function to handle the Q letter, function to handle scoring, main removes matching words on the 2 lists.
  24. ' you should be able to type 2 letters with (qu) square and get a 3 letter word.
  25.  
  26. Const TimeLimit = 180 ' actual game time is 3 minutes = 180 secs
  27. Dim Shared Board$(3, 3), WordList$(1 To 199651), OnePointList$(1 To 6955)
  28. Dim Shared As Long f48, f30, dx(7), dy(7)
  29. Dim Shared As Double BoggleTime, elapsed
  30. Dim w$, k$, player$, AI$
  31. Dim As Long printLine, pScore, aiScore, uba, ubp, i, j
  32. Screen _NewImage(800, 600, 32)
  33. _ScreenMove 200, 100
  34.     NewBoard
  35.     DisplayBoard
  36.     elapsed = 0
  37.     'Do ' display timer and allow input of words from user for 3 minutes
  38.     printLine = 20
  39.     player$ = "": w$ = ""
  40.     While elapsed < TimeLimit ' do stuff   <<<<<<<<<<<<<<<<<<<<<< off while debug wordBuildOK
  41.         Cls
  42.         DisplayBoard
  43.         elapsed = Timer(.01) - BoggleTime
  44.         If elapsed < 0 Then elapsed = 24 * 60 * 60 + Timer(.01) - BoggleTime ' midnight problem add aday of seconds to timer and subtr boogle
  45.         _Font f48
  46.         Line (300, 240)-Step(180, 60), &HFF000000, BF ' blackout last time
  47.         Color &HFFFFFF00
  48.         _PrintString (300, 240), _Trim$(Str$(TimeLimit - Int(elapsed)))
  49.  
  50.         _Font 16
  51.         Color _RGB32(200, 200, 255)
  52.         k$ = UCase$(InKey$)
  53.         If Len(k$) Then 'handle 1 and 2 char key presses, maybe replace with _keyhit later
  54.             Select Case Len(k$)
  55.                 Case 1
  56.                     Select Case Asc(k$)
  57.                         Case 3 'Ctrl + C   another way to clear?
  58.                             w$ = ""
  59.                         Case 8 ' backspace          more to do
  60.                             If Len(w$) Then w$ = Left$(w$, Len(w$) - 1)
  61.                         Case 13
  62.                             If wordBuildOK&(w$) Then
  63.                                 If Find&(WordList$(), qw$(w$)) Then ' check words before add to player$
  64.                                     If player$ = "" Then player$ = qw$(w$) Else player$ = player$ + " " + qw$(w$)
  65.                                 End If
  66.                             End If
  67.                             w$ = ""
  68.                         Case 27 'esc
  69.                             w$ = "" ': exit ?
  70.                         Case Else
  71.                             If 63 < Asc(k$) And Asc(k$) < 91 Then w$ = w$ + k$
  72.                     End Select
  73.             End Select
  74.         End If
  75.         Locate printLine, 1: Print w$
  76.         Locate printLine + 2, 1: Print player$
  77.  
  78.         _Display
  79.         _Limit 60
  80.     Wend
  81.     _PrintString (400, 240), "Times up!"
  82.  
  83.     'lets see what the AI comes up with
  84.     AI$ = AIwords$(.07) ' try 30 secs for starters, 5 still gets a complete list, try 1 sec  OK that doesn't quite finish
  85.     Locate printLine + 6, 1: Print "AI: "; AI$
  86.     Print
  87.     Print "          zzz... Press any for the time of reconning,"
  88.     Print "   matching words on 2 lists will be removed and the round scored."
  89.     Sleep
  90.     ' evalaute results (remove matching words in lists) and score
  91.     Cls
  92.     DisplayBoard
  93.     _Font 16
  94.     player$ = removeRepeats$(player$)
  95.     ReDim p(1 To 1) As String
  96.     Split player$, " ", p()
  97.     ubp = UBound(p)
  98.     ReDim a(1 To 1) As String
  99.     Split AI$, " ", a()
  100.     uba = UBound(a)
  101.     For i = 1 To uba
  102.         For j = 1 To ubp
  103.             If a(i) = p(j) Then a(i) = "": p(j) = ""
  104.         Next
  105.     Next
  106.     pScore = score&(p())
  107.     aiScore = score&(a())
  108.     AI$ = ""
  109.     player$ = ""
  110.     For i = 1 To uba
  111.         If a(i) <> "" Then AI$ = AI$ + " " + a(i)
  112.     Next
  113.     For i = 1 To ubp
  114.         If p(i) <> "" Then player$ = player$ + " " + p(i)
  115.     Next
  116.     Locate printLine + 1, 1: Print "Player:"; player$
  117.     Print " Score:"; pScore
  118.     Locate printLine + 6, 1: Print "AI:"; AI$
  119.     Print " Score:"; aiScore
  120.     Sleep
  121.  
  122. Sub NewBoard
  123.     Static BeenHere, Di$(), Numbers()
  124.     Dim As Long i, r, c
  125.  
  126.     If BeenHere = 0 Then 'load and initialize all the one time stuff
  127.         ' load fonts
  128.         f48 = _LoadFont("Arial.ttf", 48, "MONOSPACE")
  129.         f30 = _LoadFont("Arial.ttf", 30, "MONOSPACE")
  130.         If f48 <= 0 Then Print "Sub NewBoard: Font did not load, goodbye.": End
  131.  
  132.         'load abrev Dictionary ======================================== comment out while debug wordBuildOK
  133.         Open "3 to 10 Letter Words.txt" For Input As #1
  134.         Print "Loading Dictionary..."
  135.         While Not EOF(1)
  136.             i = i + 1
  137.             Input #1, WordList$(i)
  138.             'Cls: Locate 2, 1: Print i
  139.         Wend
  140.         Close #1
  141.  
  142.         ' test load of file, find last 10 items
  143.         'For i = 199651 - 10 To 199651
  144.         '    Print WordList$(i)
  145.         'Next
  146.         'Sleep    OK loading
  147.  
  148.         Open "Boggle 1 Point Words.txt" For Input As #1
  149.         Print "Loading Boggle 1 Point Words.txt for AI..."
  150.         i = 0
  151.         While Not EOF(1)
  152.             i = i + 1
  153.             Input #1, OnePointList$(i)
  154.         Wend
  155.         Close #1
  156.  
  157.         ' load dx(), dy() for testing the legality of words built from board
  158.         dx(0) = -1: dy(0) = -1 ' this is for AI to find words
  159.         dx(1) = 0: dy(1) = -1
  160.         dx(2) = 1: dy(2) = -1
  161.         dx(3) = -1: dy(3) = 0
  162.         dx(4) = 1: dy(4) = 0
  163.         dx(5) = -1: dy(5) = 1
  164.         dx(6) = 0: dy(6) = 1
  165.         dx(7) = 1: dy(7) = 1
  166.  
  167.         ' These are the 16 Dice with 6 Faces of a Letter need for Boggle
  168.         Dim Di$(0 To 15) ' this for 16 di, 6 letters per
  169.         Di$(1) = "RIFOBX"
  170.         Di$(2) = "IFEHEY"
  171.         Di$(3) = "DENOWS"
  172.         Di$(4) = "UTOKND"
  173.         Di$(5) = "HMSRAO"
  174.         Di$(6) = "LUPETS"
  175.         Di$(7) = "ACITOA"
  176.         Di$(8) = "YLGKUE"
  177.         Di$(9) = "QBMJOA"
  178.         Di$(10) = "EHISPN"
  179.         Di$(11) = "VETIGN"
  180.         Di$(12) = "BALIYT"
  181.         Di$(13) = "EZAVND"
  182.         Di$(14) = "RALESC"
  183.         Di$(15) = "UWILRG"
  184.         Di$(0) = "PACEMD"
  185.  
  186.         Dim Numbers(0 To 15) ' load numbers for shuffling die order
  187.         For i = 0 To 15
  188.             Numbers(i) = i
  189.         Next
  190.         BeenHere = -1
  191.     End If
  192.  
  193.     'now get the game going
  194.     For i = 15 To 1 Step -1 'shuffle die
  195.         Swap Numbers(i), Numbers(Int(Rnd * (i + 1)))
  196.     Next
  197.     'For i = 1 To 16: Print Numbers(i);: Next: Print   ' check the shuffle
  198.     For i = 0 To 15 'choosing random face of die = 1 Letter
  199.         Index2ColRow i, c, r
  200.         Board$(c, r) = Mid$(Di$(Numbers(i)), Int(Rnd * 6) + 1, 1)
  201.     Next
  202.     ' now set timer + 180
  203.     BoggleTime = Timer(.01)
  204.     _Font 16
  205.  
  206. Sub DisplayBoard
  207.     Dim row, col
  208.     _Font f48
  209.     For row = 0 To 3 '  display the board
  210.         For col = 0 To 3
  211.             Line ((col + 1) * 60 - 5, (row + 1) * 60 - 5)-Step(54, 54), &HFF2020FF, BF 'face color or die
  212.             If Board$(col, row) = "Q" Then 'If face has a Q it is supposed to be "Qu"
  213.                 _Font f30
  214.                 Color &HFF661111 'shade
  215.                 _PrintString ((col + 1) * 60 - 4, (row + 1) * 60 + 11), "Q"
  216.                 _PrintString ((col + 1) * 60 + 24, (row + 1) * 60 + 11), "U"
  217.                 Color &HFFBBBBBB 'letter
  218.                 _PrintString ((col + 1) * 60 - 7, (row + 1) * 60 + 9), "Q"
  219.                 _PrintString ((col + 1) * 60 + 22, (row + 1) * 60 + 9), "U"
  220.                 _Font f48
  221.             Else
  222.                 Color &HFF661111 'shade
  223.                 _PrintString ((col + 1) * 60 + 2, (row + 1) * 60 + 2), Board$(col, row)
  224.                 Color &HFFBBBBBB 'letter
  225.                 _PrintString ((col + 1) * 60, (row + 1) * 60), Board$(col, row)
  226.             End If
  227.         Next
  228.     Next
  229.  
  230. Function ColRow2Index& (row As Long, col As Long) ' convert a board letter to index (not needed yet?)
  231.     ColRow2Index& = row * 4 + col
  232. Sub Index2ColRow (indexIn As Long, rowOut As Long, colOut As Long) 'convert die index to board col, row
  233.     colOut = indexIn Mod 4: rowOut = indexIn \ 4
  234.  
  235. Sub Split (SplitMeString As String, delim As String, loadMeArray() As String)
  236.     Dim curpos As Long, arrpos As Long, LD As Long, dpos As Long 'fix use the Lbound the array already has
  237.     curpos = 1: arrpos = LBound(loadMeArray): LD = Len(delim)
  238.     dpos = InStr(curpos, SplitMeString, delim)
  239.     Do Until dpos = 0
  240.         loadMeArray(arrpos) = Mid$(SplitMeString, curpos, dpos - curpos)
  241.         arrpos = arrpos + 1
  242.         If arrpos > UBound(loadMeArray) Then ReDim _Preserve loadMeArray(LBound(loadMeArray) To UBound(loadMeArray) + 1000) As String
  243.         curpos = dpos + LD
  244.         dpos = InStr(curpos, SplitMeString, delim)
  245.     Loop
  246.     loadMeArray(arrpos) = Mid$(SplitMeString, curpos)
  247.     ReDim _Preserve loadMeArray(LBound(loadMeArray) To arrpos) As String 'get the ubound correct
  248.  
  249. Function Find& (SortedArr$(), x$) ' if I am using this only to find words in dictionary, I can mod to optimize
  250.     Dim As Long low, hi, test
  251.     low = LBound(SortedArr$): hi = UBound(SortedArr$)
  252.     While low <= hi
  253.         test = Int((low + hi) / 2)
  254.         If SortedArr$(test) = x$ Then
  255.             Find& = test: Exit Function
  256.         Else
  257.             If SortedArr$(test) < x$ Then low = test + 1 Else hi = test - 1
  258.         End If
  259.     Wend
  260.  
  261. Function wordBuildOK& (w$) ' this function checks to see that the was constructed (or is constructable with the given board).
  262.     Dim As Long r, c, test
  263.     Dim copy$(-1 To 4, -1 To 4), first$
  264.     For r = 0 To 3
  265.         For c = 0 To 3
  266.             copy$(c, r) = Board$(c, r)
  267.         Next
  268.     Next
  269.  
  270.     first$ = Mid$(w$, 1, 1)
  271.     For r = 0 To 3
  272.         For c = 0 To 3
  273.             If copy$(c, r) = first$ Then 'cell letter matches first letter in word
  274.                 test = findCell&(c, r, w$, 2, copy$())
  275.                 If test Then wordBuildOK& = -1: Exit Function ' ah ha! maybe it keeps trying when we are supposed to be done, fix?
  276.             End If
  277.         Next
  278.     Next
  279.  
  280. 'recursively called starting from wordBuildOK&
  281. Function findCell& (startX As Long, startY As Long, word$, index As Long, Arr$()) ' want to setup recursive searcher
  282.     Dim As Long d, x, y, i, r, c, test
  283.     Dim w$
  284.     'make own set of variables for this function  (attempt to debug but did not fix anything)
  285.     Dim a$(-1 To 4, -1 To 4)
  286.     For r = 0 To 3
  287.         For c = 0 To 3
  288.             a$(c, r) = Arr$(c, r)
  289.         Next
  290.     Next
  291.     i = index: w$ = word$: y = startY: x = startX
  292.     If i > Len(w$) Then findCell = -1: Exit Function
  293.     a$(x, y) = "" 'so wont be used again
  294.     For d = 0 To 7
  295.         If a$(x + dx(d), y + dy(d)) = Mid$(w$, i, 1) Then
  296.             test = findCell&(x + dx(d), y + dy(d), w$, i + 1, a$())
  297.             If test Then findCell& = -1: Exit Function
  298.         End If
  299.     Next
  300.  
  301. Function AIwords$ (timeLimit As Double) 'returns a space delimiter string of 1 point words that can be constructed from board in limited time
  302.     Dim As Double startTime, checkTime
  303.     Dim As Long i, r, c, OK, dp, l, ub
  304.     Dim l$, letters$, b$
  305.     startTime = Timer(.01)
  306.     ub = UBound(OnePointList$)
  307.     ' get a non redundant list of letters from board
  308.     For r = 0 To 3
  309.         For c = 0 To 3
  310.             l$ = Board$(c, r)
  311.             If (r = 0) And (c = 0) Then
  312.                 letters$ = l$
  313.             Else
  314.                 If InStr(letters$, l$) <= 0 Then '  insrt letter
  315.                     OK = 0
  316.                     For i = 1 To Len(letters$)
  317.                         If Asc(l$) < Asc(letters$, i) Then ' insert spotted
  318.                             If i = 1 Then
  319.                                 letters$ = l$ + letters$: OK = -1: Exit For
  320.                             Else
  321.                                 letters$ = Mid$(letters$, 1, i - 1) + l$ + Mid$(letters$, i)
  322.                                 OK = -1: Exit For
  323.                             End If
  324.                         End If
  325.                     Next
  326.                     If OK = 0 Then letters$ = letters$ + l$
  327.                 End If
  328.             End If
  329.         Next
  330.     Next
  331.     'check if this is OK so far  OK finally!  This is 3rd time I needed to exit when found
  332.     ' AIwords$ = letters$
  333.     'now letters of board are in alpha order
  334.     dp = 1 'place in dict
  335.     For l = 1 To Len(letters$) ' advance place in list$ by one until the word > letter
  336.         While Asc(OnePointList$(dp), 1) < Asc(letters$, l)
  337.             dp = dp + 1
  338.             If dp > ub Then GoTo fini
  339.             If Timer(.01) - startTime < 0 Then checkTime = Timer(.01) + 24 * 60 * 60 Else checkTime = Timer(.01)
  340.             If checkTime - startTime > timeLimit Then GoTo fini
  341.         Wend
  342.         'now start testing words
  343.         While Asc(OnePointList$(dp), 1) = Asc(letters$, l)
  344.             If wordBuildOK&(OnePointList$(dp)) Then
  345.                 If b$ = "" Then b$ = OnePointList$(dp) Else b$ = b$ + " " + OnePointList$(dp)
  346.             End If
  347.             dp = dp + 1
  348.             If dp > ub Then GoTo fini
  349.             If Timer(.01) - startTime < 0 Then checkTime = Timer(.01) + 24 * 60 * 60 Else checkTime = Timer(.01)
  350.             If checkTime - startTime > timeLimit Then GoTo fini
  351.         Wend
  352.     Next
  353.  
  354.     fini:
  355.     AIwords$ = b$
  356.  
  357. Function qw$ (w$) 'insert the u into a q letter word
  358.     Dim As Long p
  359.     p = InStr(w$, "Q")
  360.     If p Then qw$ = Mid$(w$, 1, p) + "U" + Mid$(w$, p + 1) Else qw$ = w$
  361.  
  362. Function removeRepeats$ (s$) ' s$ is space delimited word list
  363.     ReDim t$(1 To 1), b$
  364.     Dim As Long ub, i, j, ok
  365.     Split s$, " ", t$()
  366.     ub = UBound(t$)
  367.     For i = 1 To ub
  368.         ok = -1
  369.         For j = 1 To i - 1
  370.             If t$(i) = t$(j) Then ok = 0: Exit For
  371.         Next
  372.         If ok Then
  373.             If b$ = "" Then b$ = t$(i) Else b$ = b$ + " " + t$(i)
  374.         End If
  375.     Next
  376.     removeRepeats$ = b$
  377.  
  378. Function score& (a() As String)
  379.     Dim As Long i, s
  380.     For i = 1 To UBound(a)
  381.         Select Case Len(a(i))
  382.             Case 3, 4: s = s + 1
  383.             Case 5: s = s + 2
  384.             Case 6: s = s + 3
  385.             Case 7: s = s + 5
  386.             Case Is > 7: s = s + 11
  387.         End Select
  388.     Next
  389.     score& = s
  390.  

 


 


Dang, I missed god.

Title: Re: Boggle play against AI - WIP
Post by: SMcNeill on January 15, 2022, 01:08:04 pm
Why doesn't the AI find PEAT and DIE and TAP?

It's overlooking a lot of easy words.
Title: Re: Boggle play against AI - WIP
Post by: bplus on January 15, 2022, 01:56:51 pm
Why doesn't the AI find PEAT and DIE and TAP?

It's overlooking a lot of easy words.

They were on my list ("a lot of easy words"), at time of reckoning matching words are eliminated and then words remaining are scored.

You see those words on the screen shot before the last (when I had words in my list). The AI of course found every 3-4 letter word I did (wiping out my list) plus a couple more, it gets to keep the couple more :)

I will add a label about typing in the words while the timer is going, but I have bigger fish to fry. I want that 17th die to eliminate special handling of Qwords that really threw a monkey wrench in coding this game and AI's list will have access to ALL 3 and 4 letter words. There many q words that don't use u after the q, why should we shun them?

We play games but we also expand vocabulary, spelling and data processing skills ie keep our minds active and growing.
Title: Re: Boggle play against AI - WIP
Post by: jack on January 15, 2022, 02:46:59 pm
Hi bplus
about the program simply vanishing while reading the dictionary, try Dim WordList$(1 To 300000) even though the number of words is only 279499
but where can I find "Boggle 1 Point Words.txt" ?
Title: Re: Boggle play against AI - WIP
Post by: bplus on January 15, 2022, 03:22:33 pm
Hi bplus
about the program simply vanishing while reading the dictionary, try Dim WordList$(1 To 300000) even though the number of words is only 279499
but where can I find "Boggle 1 Point Words.txt" ?

Apologies to all, I forgot you guys need the files to play, dang sorry.

Use the link for the giant Collins Scrabble Word (2019).txt files, I would get the dictionary with the word list to learn what some of these crazy 3-4 letter words are!? You can find the link right under the Boggle Title line at the start of the bas source of Boggle 1.

And here is the zip for the Bas Make codes ran to create smaller files from the Collins word list.

+1) The 3 to 10 Letter Words.txt is my fix for being unable to load the entire Collins Word List as mention in reply above. Might be able to get more Letter words thn 10 but I roughing out a running game not optimizing.
This file was used to load most of the words into an array in the app, as all the words wont fit.

+2) The Boggle 1 Point Words.txt are what the AI uses to attempt legal constructions from the given active game board. One point words only have 3 or 4 Letters.

So in the zip you have the txt file lists and the code that built them from the Collins files and the Boggle 1 source, but not the Collins files.
 

Thanks @jack for bringing this to my attention.


You know since the frick'n AI is so good at finding words, I think I will restrict it to 5 letters or more and give all the one pointers the player finds to the player. That will occupy the AI for finding longer words (and getting bigger points unless player finds them too.)

That may make a game the player could win! ;-)) and it would be more interesting checking out what the AI finds.
Title: Re: Boggle play against AI - WIP
Post by: SMcNeill on January 16, 2022, 05:28:39 am
Give the AI an education level that the user can select from 1 to 10.

1 has a 10% chance to "know" a word if it finds a match; 10 has a 100% chance.

For example, the AI finds "cat", but with the handicap, it only has a 10 * level percent chance of keeping the word in its list.

Seems simple enough to implement and allows for some customization for game difficulty.
Title: Re: Boggle play against AI - WIP
Post by: bplus on January 16, 2022, 08:46:04 am
Give the AI an education level that the user can select from 1 to 10.

1 has a 10% chance to "know" a word if it finds a match; 10 has a 100% chance.

For example, the AI finds "cat", but with the handicap, it only has a 10 * level percent chance of keeping the word in its list.

Seems simple enough to implement and allows for some customization for game difficulty.

Ha! yeah do something? I tried the 5 or more letters to AI only taking away all 3-4 letter words, it still beats the crap out of the player (me) with higher point words!

I gotta show all the words that can be found on the board, it is incredible!

BTW I got the special handling of Q words removed added the 17th dice and set it up to click the letters and right click to enter the word through the legalConstructionOK function and the Dictionary check as with the keyboard inputs.
Title: Re: Boggle play against AI - WIP
Post by: _vince on January 16, 2022, 12:33:02 pm
bplus is into letters now instead of graphics, sad times
Title: Re: Boggle play against AI - WIP
Post by: bplus on January 16, 2022, 12:34:16 pm
 Boggle 2 update with Qwords special handling gone, click letters and Right click to "Enter" word, one word list load that AI has full access to so we can see what we could't when searching for words ourself. I could only get a list of 3 to 12 letter Words to load and last night that quit working for some unknown reason. If that happens to you, I left a smaller file of 3 to 10 letter words to load mod the Sub NewBoard that initializes the game:
Code: QB64: [Select]
  1. _Title "Boggle 2" ' b+ start 2022-01-12
  2. ' Scabble Word List and Dictionary
  3. ' ref dictionary: https://boardgames.stackexchange.com/questions/38366/latest-collins-scrabble-words-list-in-text-file
  4. ' Die configurations
  5. ' https://boardgames.stackexchange.com/questions/29264/boggle-what-is-the-dice-configuration-for-boggle-in-various-languages
  6. ' Thank you!
  7.  
  8. ' 2022-01-14 status have the board display separated out from New Board and Game initialization
  9. ' have timer system and display working next to board display
  10. ' have player input system working
  11. '   1. check word in dictionary working
  12. '   2. check that word can legally be built from board is funky! sometimes works sometimes not 50/50 for legit builds yuck
  13. '     How to debug? Todays task is get this abolutely critical function working so the AI would be fairly easy to do.
  14. ' Oh I had to modify the Collins Word list to get a size that would fit into a variable length string array, 279496 words
  15. ' kept crashing QB64 without error messages when reached about 260,000 words so now I have a list for words with lengths of
  16. ' between 3 and 10, still 199,651 words!
  17. ' 2202-01-14 Hurray! Function wordBuildOK& is fixed, a very simple logic bug. I continued to check for builds of word after
  18. ' I got confirmation one of the builds was possible. It kept checking if there was more than one place the first letter of word
  19. ' appeared. The function is crucial for AI testing if words are buildable. I can get started on that today now that bug is
  20. ' fixed. Man! I had this fixed almost immediately, I had to exit the Function when the recursive function findCell& called from
  21. ' it found a positive result.
  22.  
  23. ' 2022-01-15 qw$() function to handle the Q letter, function to handle scoring, main removes matching words on the 2 lists.
  24. ' you should be able to type 2 letters with (qu) square and get a 3 letter word.
  25.  
  26. ' 2022-01-15 screw the die with face Q(u) with implied u, adding a 17th die with AEIOUU and removing all the special handling
  27. ' for that implied u crap! I don't think this will ruin the game but only make it better and less difficlut to code.
  28. ' Fix the spelling of reckoning :)  OK new word files made for Boggle 2: AI Check Words is HUGE! Almost 35 X's more and longer
  29. ' words, words 5 letters and longer. The Word List file has been expanded from 3 to 10 letter words to 3 to 12 letter words.
  30. ' Dang! It's worse the points the AI collects from big words times the point values they are worth...
  31. ' OK more handicap, AI concedes all 3-6 letters words to player and gets points only for 7+ letter words. Is it fair now?
  32. ' Too much! Try concedes 3-5 but fight for 6+ letter words.
  33.  
  34. ' 2022-01-16 Yesterday I was able to use 2 large word lists from : 3 - 12 Letter Words.txt (249,702)
  35. ' and AI Check Words.txt (229,745) they loaded fine all afternoon. Then I decided to use the 3-12 letter words for the
  36. ' AI check words list so I only had to load one file! So I change the code to do that and suddenly the 3-12 Letter Words.txt
  37. ' was quitting like there were more words than it could load (no error message, no subscript out of range you get with an array
  38. ' dim'd too small for # of items to put in it), QB64 just bugged out and quit ??? Just like when I was trying to load the
  39. ' whole file.
  40. '    So today I m trying jack's idea to over size the dim of the array to load into and track the number going in with a
  41. ' variable NTopWord. And I am going to try and load a bigger 3-16 Letter Words.txt file. NOPE!
  42. ' The 3-12 list is loading again today, but no garantee it will work for others or me in future as it quit last night.
  43. ' Try loading up Browser and run other stuff, maybe QB64 variable length string loads in one array is limited. Still OK.
  44. ' Have no idea why it works for .5 day and later doesn't ????
  45.  
  46. ' 2022-01-16 Boggle 2 post: I will have the 3 to 10 Letter Words.txt file to load into game if you or I have problems loading the
  47. ' 3 to 12 Letter Words.txt file. So just Boggle 2 and 2 word files in zip.
  48.  
  49. '  =================================== Sound Signals =========================================================
  50. ' Oh a high pitched peep is a sound signalling that the word enetered can't be legally built from board.
  51. ' A Beep is a signal the word is not in dictionary.
  52.  
  53. Const TimeLimit = 180 ' actual game time is 3 minutes = 180 secs
  54. Const PrintLine = 20 ' row where I can start display of user inputs below Board display on screen.
  55. Dim Shared Board$(3, 3), WordList$(1 To 280000) ' < this is over sized array # words is 279369 try jacks idea of oversized array
  56. Dim Shared As Long f48, f30, dx(7), dy(7), NTopWord
  57. Dim Shared As Double BoggleTime, elapsed
  58. Dim w$, k$, player$, AI$
  59. Dim As Long pScore, aiScore, uba, ubp, i, j, mx, my, mbL, mbR, row, col
  60. Screen _NewImage(800, 600, 32)
  61. _ScreenMove 200, 100
  62.  
  63. ' quick board for testin working mouse locations for clicking letters
  64. 'NewBoard
  65. 'While _KeyDown(27) = 0
  66. '    Cls
  67. '    DisplayBoard
  68. '    _Font 16
  69. '    While _MouseInput: Wend
  70. '    mx = _MouseX: my = _MouseY: mbL = _MouseButton(1): mbR = _MouseButton(2)
  71. '    row = (my - 80) / 60: col = (mx - 80) / 60 ' ?? not -60 but more
  72. '    Print col, row, mbL, mbR
  73. '    _Display
  74. '    _Limit 60
  75. 'Wend
  76.  
  77.     NewBoard ' display timer and allow input of words from user for 3 minutes
  78.     DisplayBoard
  79.     elapsed = 0: player$ = "": w$ = ""
  80.     While elapsed < TimeLimit ' do stuff   <<<<<<<<<<<<<<<<<<<<<< off while debug wordBuildOK
  81.         Cls
  82.         DisplayBoard
  83.         elapsed = Timer(.01) - BoggleTime
  84.         If elapsed < 0 Then elapsed = 24 * 60 * 60 + Timer(.01) - BoggleTime ' midnight problem add aday of seconds to timer and subtr boogle
  85.         _Font f48
  86.         Color &HFFFFFF00
  87.         _PrintString (300, 240), _Trim$(Str$(TimeLimit - Int(elapsed)))
  88.  
  89.         _Font 16
  90.         Color _RGB32(200, 200, 255)
  91.  
  92.         'some help
  93.         Locate 5, 50: Print "Use Keyboard or Mouse to add letters to word."
  94.         Locate 6, 50: Print " Backspace will clear last letter."
  95.         Locate 7, 50: Print " Esc or Ctrl+C will clear word started."
  96.         Locate 8, 50: Print " Enter or Right Mouse to checks word > list."
  97.  
  98.         'add mouse controls
  99.         While _MouseInput: Wend
  100.         mx = _MouseX: my = _MouseY: mbL = _MouseButton(1): mbR = _MouseButton(2)
  101.         row = (my - 80) / 60: col = (mx - 80) / 60 ' ?? not -60 but more
  102.         If mbL Then
  103.             _Delay .25
  104.             If -1 < row And row < 4 Then
  105.                 If -1 < col And col < 4 Then
  106.                     w$ = w$ + Board$(col, row) ' add letter to w$ build
  107.                 End If
  108.             End If
  109.         End If
  110.         If mbR Then ' same as hitting Enter on keyboard
  111.             _Delay .25
  112.             If wordBuildOK&(w$) Then ' legal build
  113.                 If Find&(WordList$(), w$) Then ' check words in dictionary before add to player$
  114.                     If player$ = "" Then player$ = w$ Else player$ = player$ + " " + w$
  115.                 Else
  116.                     Beep
  117.                 End If
  118.             Else
  119.                 Sound 3000, 1
  120.             End If
  121.             w$ = ""
  122.         End If
  123.  
  124.         k$ = UCase$(InKey$) ' compare to all caps
  125.         If Len(k$) Then 'handle 1 and 2 char key presses, maybe replace with _keyhit later
  126.             Select Case Len(k$)
  127.                 Case 1
  128.                     Select Case Asc(k$)
  129.                         Case 3 'Ctrl + C   another way to clear?
  130.                             w$ = ""
  131.                         Case 8 ' backspace          more to do
  132.                             If Len(w$) Then w$ = Left$(w$, Len(w$) - 1)
  133.                         Case 13
  134.                             If wordBuildOK&(w$) Then ' legal build
  135.                                 If Find&(WordList$(), w$) Then ' check words in dictionary before add to player$
  136.                                     If player$ = "" Then player$ = w$ Else player$ = player$ + " " + w$
  137.                                 Else
  138.                                     Beep
  139.                                 End If
  140.                             Else
  141.                                 Sound 3000, 1
  142.                             End If
  143.                             w$ = ""
  144.                         Case 27 'esc
  145.                             w$ = "" ': exit ?
  146.                         Case Else
  147.                             If 63 < Asc(k$) And Asc(k$) < 91 Then w$ = w$ + k$
  148.                     End Select
  149.             End Select
  150.         End If
  151.         Locate PrintLine, 1: Print w$
  152.         Locate PrintLine + 1, 1: Print player$
  153.         _Display
  154.         _Limit 60
  155.     Wend
  156.     _PrintString (400, 240), "Times up!"
  157.  
  158.     'lets see what the AI comes up with
  159.     AI$ = AIwords$(5) ' new much larger much longer words
  160.     Locate PrintLine + 4, 1: Print "AI: "; AI$
  161.     Print "          zzz... Press any for the time of reckoning,"
  162.     Print "   matching words on 2 lists will be removed and the round scored."
  163.     Sleep
  164.     ' evalaute results (remove matching words in lists) and score
  165.     Cls
  166.     DisplayBoard
  167.     _Font 16
  168.     player$ = removeRepeats$(player$) ' remove words input 2 or more times
  169.     ReDim p(1 To 1) As String 'setup to convert to an array
  170.     Split player$, " ", p() ' convert player$ to array
  171.     ubp = UBound(p) ' quick count of words because array is base 1
  172.  
  173.     ReDim a(1 To 1) As String ' likewise for AI's string list
  174.     Split AI$, " ", a()
  175.     uba = UBound(a)
  176.  
  177.     For i = 1 To uba ' Remove matching words from lists
  178.         For j = 1 To ubp
  179.             If a(i) = p(j) Then a(i) = "": p(j) = ""
  180.         Next
  181.     Next
  182.     pScore = score&(p()): aiScore = score&(a()) ' scores done
  183.  
  184.     AI$ = "": player$ = "" ' reset strings for rebuild
  185.     For i = 1 To uba ' rebuild AI's non matching words
  186.         If a(i) <> "" Then AI$ = AI$ + " " + a(i)
  187.     Next
  188.     For i = 1 To ubp ' rebuild players non matching words (if any)
  189.         If p(i) <> "" Then player$ = player$ + " " + p(i)
  190.     Next
  191.     'display results of reckoning for round
  192.     Locate PrintLine, 1: Print "Player:"; player$; " Score:"; pScore
  193.     Locate PrintLine + 3, 1: Print "AI:"; AI$; " Score:"; aiScore
  194.     Print "      zzz... read em and weep, press any for another round of humiliation :)"
  195.     Sleep
  196.  
  197. Sub NewBoard
  198.     Static BeenHere, Di$(), Numbers()
  199.     Dim As Long i, r, c
  200.  
  201.     If BeenHere = 0 Then 'load and initialize all the one time stuff
  202.         ' load fonts
  203.         f48 = _LoadFont("Arial.ttf", 48, "MONOSPACE")
  204.         f30 = _LoadFont("Arial.ttf", 30, "MONOSPACE")
  205.         If f48 <= 0 Then Print "Sub NewBoard: Font did not load, goodbye.": End
  206.  
  207.         'load abrev Dictionary ======================================== comment out while debug
  208.         Open "3 to 12 Letter Words.txt" For Input As #1 ' for some reason 3-12 stopped loading on me
  209.         '  "3 to 16 Letter Words.txt" new bigger file than loaded
  210.         ' jack idea didn't work
  211.         ' lets see if 3 to 12 will load this morning, yes it's loading this morning (Browser off)
  212.  
  213.         Print "Loading 3 to 12 Letter Words from Collins Dictionary..."
  214.         While Not EOF(1)
  215.             i = i + 1
  216.             Input #1, WordList$(i)
  217.             'Cls: Locate 2, 1: Print i
  218.         Wend
  219.         Close #1
  220.         NTopWord = i
  221.         Print "Big Word File loaded!,"; NTopWord; " items   zzz..."
  222.         Sleep
  223.  
  224.         ' test load of file, find last 10 items
  225.         'For i = 199651 - 10 To 199651  ' when testing 3 to 10 Letter Words.txt
  226.         '    Print WordList$(i)
  227.         'Next
  228.         'Sleep    OK loading
  229.  
  230.         ' When using a separate word file for AI to "Handicap" it, no good just adds time to load stuff
  231.         'Open "AI Check Words.txt" For Input As #1
  232.         'Print "Loading AI Check Words.txt for AI, 6 to 12 letter words same dictionary..."
  233.         'i = 0
  234.         'While Not EOF(1)
  235.         '    i = i + 1
  236.         '    Input #1, AICheckWords$(i)
  237.         'Wend
  238.         'Close #1
  239.  
  240.         ' load dx(), dy() for testing the legality of words built from board
  241.         dx(0) = -1: dy(0) = -1 ' this is for AI to find words
  242.         dx(1) = 0: dy(1) = -1
  243.         dx(2) = 1: dy(2) = -1
  244.         dx(3) = -1: dy(3) = 0
  245.         dx(4) = 1: dy(4) = 0
  246.         dx(5) = -1: dy(5) = 1
  247.         dx(6) = 0: dy(6) = 1
  248.         dx(7) = 1: dy(7) = 1
  249.  
  250.         ' These are the 17 Dice with 6 Faces of a Letter need for Boggle
  251.         Dim Di$(0 To 16)
  252.         Di$(0) = "PACEMD"
  253.         Di$(1) = "RIFOBX"
  254.         Di$(2) = "IFEHEY"
  255.         Di$(3) = "DENOWS"
  256.         Di$(4) = "UTOKND"
  257.         Di$(5) = "HMSRAO"
  258.         Di$(6) = "LUPETS"
  259.         Di$(7) = "ACITOA"
  260.         Di$(8) = "YLGKUE"
  261.         Di$(9) = "QBMJOA"
  262.         Di$(10) = "EHISPN"
  263.         Di$(11) = "VETIGN"
  264.         Di$(12) = "BALIYT"
  265.         Di$(13) = "EZAVND"
  266.         Di$(14) = "RALESC"
  267.         Di$(15) = "UWILRG"
  268.         Di$(16) = "AEIOUU" ' b+ mod Boggle 2 to remove all special handling of Q words!!!
  269.  
  270.         Dim Numbers(0 To 16) ' load numbers for shuffling die order
  271.         For i = 0 To 16
  272.             Numbers(i) = i
  273.         Next
  274.         BeenHere = -1
  275.     End If
  276.  
  277.     'now get the game going
  278.     For i = 16 To 1 Step -1 'shuffle die
  279.         Swap Numbers(i), Numbers(Int(Rnd * (i + 1)))
  280.     Next
  281.     'For i = 1 To 16: Print Numbers(i);: Next: Print   ' check the shuffle
  282.     For i = 0 To 15 'choosing random face of die = 1 Letter
  283.         Index2ColRow i, c, r
  284.         Board$(c, r) = Mid$(Di$(Numbers(i)), Int(Rnd * 6) + 1, 1) ' one die gets is left out now Boggle 2
  285.     Next
  286.     ' now set timer + 180
  287.     BoggleTime = Timer(.01)
  288.     _Font 16
  289.  
  290. Sub DisplayBoard
  291.     Dim row, col
  292.     _Font f48
  293.     For row = 0 To 3 '  display the board
  294.         For col = 0 To 3
  295.             Line ((col + 1) * 60 - 5, (row + 1) * 60 - 5)-Step(54, 54), &HFF2020FF, BF 'face color or die
  296.             Color &HFF661111 'shade
  297.             _PrintString ((col + 1) * 60 + 2, (row + 1) * 60 + 2), Board$(col, row)
  298.             Color &HFFBBBBBB 'letter
  299.             _PrintString ((col + 1) * 60, (row + 1) * 60), Board$(col, row)
  300.         Next
  301.     Next
  302.  
  303. 'might not need in final cut
  304. Function ColRow2Index& (row As Long, col As Long) ' convert a board letter to index (not needed yet?)
  305.     ColRow2Index& = row * 4 + col
  306.  
  307. Sub Index2ColRow (indexIn As Long, rowOut As Long, colOut As Long) 'convert die index to board col, row
  308.     colOut = indexIn Mod 4: rowOut = indexIn \ 4
  309.  
  310. Sub Split (SplitMeString As String, delim As String, loadMeArray() As String)
  311.     Dim curpos As Long, arrpos As Long, LD As Long, dpos As Long 'fix use the Lbound the array already has
  312.     curpos = 1: arrpos = LBound(loadMeArray): LD = Len(delim)
  313.     dpos = InStr(curpos, SplitMeString, delim)
  314.     Do Until dpos = 0
  315.         loadMeArray(arrpos) = Mid$(SplitMeString, curpos, dpos - curpos)
  316.         arrpos = arrpos + 1
  317.         If arrpos > UBound(loadMeArray) Then ReDim _Preserve loadMeArray(LBound(loadMeArray) To UBound(loadMeArray) + 1000) As String
  318.         curpos = dpos + LD
  319.         dpos = InStr(curpos, SplitMeString, delim)
  320.     Loop
  321.     loadMeArray(arrpos) = Mid$(SplitMeString, curpos)
  322.     ReDim _Preserve loadMeArray(LBound(loadMeArray) To arrpos) As String 'get the ubound correct
  323.  
  324. Function Find& (SortedArr$(), x$) ' if I am using this only to find words in dictionary, I can mod to optimize
  325.     Dim As Long low, hi, test
  326.     low = LBound(SortedArr$): hi = UBound(SortedArr$)
  327.     While low <= hi
  328.         test = Int((low + hi) / 2)
  329.         If SortedArr$(test) = x$ Then
  330.             Find& = test: Exit Function
  331.         Else
  332.             If SortedArr$(test) < x$ Then low = test + 1 Else hi = test - 1
  333.         End If
  334.     Wend
  335.  
  336. ' This function checks to see that the word w$ is legally constructable with the given board.
  337. ' This function requires the recurive Function findCell& (startX As Long, startY As Long, word$, index As Long, Arr$())
  338. Function wordBuildOK& (w$)
  339.     Dim As Long r, c, test
  340.     Dim copy$(-1 To 4, -1 To 4), first$
  341.     For r = 0 To 3
  342.         For c = 0 To 3
  343.             copy$(c, r) = Board$(c, r)
  344.         Next
  345.     Next
  346.     first$ = Mid$(w$, 1, 1)
  347.     For r = 0 To 3
  348.         For c = 0 To 3
  349.             If copy$(c, r) = first$ Then 'cell letter matches first letter in word
  350.                 test = findCell&(c, r, w$, 2, copy$())
  351.                 If test Then wordBuildOK& = -1: Exit Function ' ah ha! maybe it keeps trying when we are supposed to be done, fix?
  352.             End If
  353.         Next
  354.     Next
  355.  
  356. 'recursively called starting from wordBuildOK&
  357. Function findCell& (startX As Long, startY As Long, word$, index As Long, Arr$()) ' want to setup recursive searcher
  358.     Dim As Long d, x, y, i, r, c, test
  359.     Dim w$
  360.     'make own set of variables for this function  (attempt to debug but did not fix anything)
  361.     Dim a$(-1 To 4, -1 To 4)
  362.     For r = 0 To 3
  363.         For c = 0 To 3
  364.             a$(c, r) = Arr$(c, r)
  365.         Next
  366.     Next
  367.     i = index: w$ = word$: y = startY: x = startX
  368.     If i > Len(w$) Then findCell = -1: Exit Function
  369.     a$(x, y) = "" 'so wont be used again
  370.     For d = 0 To 7
  371.         If a$(x + dx(d), y + dy(d)) = Mid$(w$, i, 1) Then
  372.             test = findCell&(x + dx(d), y + dy(d), w$, i + 1, a$())
  373.             If test Then findCell& = -1: Exit Function
  374.         End If
  375.     Next
  376.  
  377. Function AIwords$ (timeLimit As Double) 'returns a space delimiter string of 1 point words that can be constructed from board in limited time
  378.     Dim As Double startTime, checkTime
  379.     Dim As Long i, r, c, OK, dp, l
  380.     Dim l$, letters$, b$
  381.     startTime = Timer(.01)
  382.     'ub = UBound(WordList$)  now is NTopWord
  383.     ' get a non redundant list of letters from board and put them in alpha order
  384.     For r = 0 To 3
  385.         For c = 0 To 3
  386.             l$ = Board$(c, r)
  387.             If (r = 0) And (c = 0) Then
  388.                 letters$ = l$
  389.             Else
  390.                 If InStr(letters$, l$) <= 0 Then '  insrt letter
  391.                     OK = 0
  392.                     For i = 1 To Len(letters$) '            where?
  393.                         If Asc(l$) < Asc(letters$, i) Then ' here!
  394.                             letters$ = Mid$(letters$, 1, i - 1) + l$ + Mid$(letters$, i)
  395.                             OK = -1: Exit For
  396.                         End If
  397.                     Next
  398.                     If OK = 0 Then letters$ = letters$ + l$
  399.                 End If
  400.             End If
  401.         Next
  402.     Next
  403.     'check if this is OK so far  OK finally!  This is 3rd time I needed to exit when found
  404.     ' AIwords$ = letters$
  405.     'now letters of board are in alpha order
  406.     dp = 1 'place in dict
  407.     For l = 1 To Len(letters$) ' advance place in list$ by one until the word > letter
  408.         While Asc(WordList$(dp), 1) < Asc(letters$, l)
  409.             dp = dp + 1
  410.             If dp > NTopWord Then GoTo fini
  411.             If Timer(.01) - startTime < 0 Then checkTime = Timer(.01) + 24 * 60 * 60 Else checkTime = Timer(.01)
  412.             If checkTime - startTime > timeLimit Then GoTo fini
  413.         Wend
  414.         'now start testing words
  415.         While Asc(WordList$(dp), 1) = Asc(letters$, l)
  416.             If wordBuildOK&(WordList$(dp)) Then
  417.                 If b$ = "" Then b$ = WordList$(dp) Else b$ = b$ + " " + WordList$(dp)
  418.             End If
  419.             dp = dp + 1
  420.             If dp > NTopWord Then GoTo fini
  421.             If Timer(.01) - startTime < 0 Then checkTime = Timer(.01) + 24 * 60 * 60 Else checkTime = Timer(.01)
  422.             If checkTime - startTime > timeLimit Then GoTo fini
  423.         Wend
  424.     Next
  425.  
  426.     fini:
  427.     AIwords$ = b$
  428.  
  429. ' Dont need to special handling of Q words
  430. 'Function qw$ (w$) 'insert the u into a q letter word
  431. '    Dim As Long p
  432. '    p = InStr(w$, "Q")
  433. '    If p Then qw$ = Mid$(w$, 1, p) + "U" + Mid$(w$, p + 1) Else qw$ = w$
  434. 'End Function
  435.  
  436. Function removeRepeats$ (s$) ' s$ is space delimited word list
  437.     ReDim t$(1 To 1), b$
  438.     Dim As Long ub, i, j, ok
  439.     Split s$, " ", t$()
  440.     ub = UBound(t$)
  441.     For i = 1 To ub
  442.         ok = -1
  443.         For j = 1 To i - 1
  444.             If t$(i) = t$(j) Then ok = 0: Exit For
  445.         Next
  446.         If ok Then
  447.             If b$ = "" Then b$ = t$(i) Else b$ = b$ + " " + t$(i)
  448.         End If
  449.     Next
  450.     removeRepeats$ = b$
  451.  
  452. Function score& (a() As String)
  453.     Dim As Long i, s
  454.     For i = 1 To UBound(a)
  455.         Select Case Len(a(i))
  456.             Case 3, 4: s = s + 1
  457.             Case 5: s = s + 2
  458.             Case 6: s = s + 3
  459.             Case 7: s = s + 5
  460.             Case Is > 7: s = s + 11
  461.         End Select
  462.     Next
  463.     score& = s
  464.  

Boggle 2 Before Reckoning:
 


Boggle 2 Post Reckoning:
 


Zip contains 2 txt files of word lists and Boggle 2.bas source, if 3 to 12 Letter Words.txt file does not load, mod the Sub NewBoard to try the 3 to 10 Letter Words.txt file.

My lesson from Boggle 2 is that the AI is too good for any decent challenge between Human and machine, so I will try to use this code to create a Vocabulary and Spelling skills builder program.

Oh! interesting note, for some reason the code would not accept my attempts (yes more than once) of YELL, I wanted to YELL! I cant remember getting a ill-legal build signal or a not found in dictionary signal.

BTW Boggle 2 does signal why it doesn't allow a word into you accumulating list with sounds a high pitch one is not buildable from board and a BEEP is a not found in dictionary signal.

Title: Re: Boggle play against AI - WIP
Post by: bplus on January 16, 2022, 12:39:49 pm
bplus is into letters now instead of graphics, sad times

ehhhh... been doing words for awhile.

I am not best with vocabulary and spelling (or reading for that matter) but maybe I can improve through games?

@_vince you are doing pretty well with the graphics yourself! was it you or Cobalt working on math (or both).

QB64 can make learning fun with creative challenges!
Title: Re: Boggle play against AI - WIP
Post by: bplus on January 16, 2022, 08:51:52 pm
Solved the problem with loading all the words. Now I load none! I converted the 2 Collins files to RA and now my arrays are on disk. Don't really notice less speed, the AI can search the whole word list just fine in a second.

Now after the reckoning you can look up words by pressing h (help with word) instead of any other key.
  [ This attachment cannot be displayed inline in 'Print Page' view ]  

Code: QB64: [Select]
  1. _Title "Boggle 3" ' b+ start 2022-01-12
  2. ' Scabble Word List and Dictionary
  3. ' ref dictionary: https://boardgames.stackexchange.com/questions/38366/latest-collins-scrabble-words-list-in-text-file
  4. ' Die configurations
  5. ' https://boardgames.stackexchange.com/questions/29264/boggle-what-is-the-dice-configuration-for-boggle-in-various-languages
  6. ' Thank you!
  7.  
  8. ' 2022-01-14 status have the board display separated out from New Board and Game initialization
  9. ' have timer system and display working next to board display
  10. ' have player input system working
  11. '   1. check word in dictionary working
  12. '   2. check that word can legally be built from board is funky! sometimes works sometimes not 50/50 for legit builds yuck
  13. '     How to debug? Todays task is get this abolutely critical function working so the AI would be fairly easy to do.
  14. ' Oh I had to modify the Collins Word list to get a size that would fit into a variable length string array, 279496 words
  15. ' kept crashing QB64 without error messages when reached about 260,000 words so now I have a list for words with lengths of
  16. ' between 3 and 10, still 199,651 words!
  17. ' 2202-01-14 Hurray! Function wordBuildOK& is fixed, a very simple logic bug. I continued to check for builds of word after
  18. ' I got confirmation one of the builds was possible. It kept checking if there was more than one place the first letter of word
  19. ' appeared. The function is crucial for AI testing if words are buildable. I can get started on that today now that bug is
  20. ' fixed. Man! I had this fixed almost immediately, I had to exit the Function when the recursive function findCell& called from
  21. ' it found a positive result.
  22.  
  23. ' 2022-01-15 qw$() function to handle the Q letter, function to handle scoring, main removes matching words on the 2 lists.
  24. ' you should be able to type 2 letters with (qu) square and get a 3 letter word.
  25.  
  26. ' 2022-01-15 screw the die with face Q(u) with implied u, adding a 17th die with AEIOUU and removing all the special handling
  27. ' for that implied u crap! I don't think this will ruin the game but only make it better and less difficlut to code.
  28. ' Fix the spelling of reckoning :)  OK new word files made for Boggle 2: AI Check Words is HUGE! Almost 35 X's more and longer
  29. ' words, words 5 letters and longer. The Word List file has been expanded from 3 to 10 letter words to 3 to 12 letter words.
  30. ' Dang! It's worse the points the AI collects from big words times the point values they are worth...
  31. ' OK more handicap, AI concedes all 3-6 letters words to player and gets points only for 7+ letter words. Is it fair now?
  32. ' Too much! Try concedes 3-5 but fight for 6+ letter words.
  33.  
  34. ' 2022-01-16 Yesterday I was able to use 2 large word lists from : 3 - 12 Letter Words.txt (249,702)
  35. ' and AI Check Words.txt (229,745) they loaded fine all afternoon. Then I decided to use the 3-12 letter words for the
  36. ' AI check words list so I only had to load one file! So I change the code to do that and suddenly the 3-12 Letter Words.txt
  37. ' was quitting like there were more words than it could load (no error message, no subscript out of range you get with an array
  38. ' dim'd too small for # of items to put in it), QB64 just bugged out and quit ??? Just like when I was trying to load the
  39. ' whole file.
  40. '    So today I m trying jack's idea to over size the dim of the array to load into and track the number going in with a
  41. ' variable NTopWord. And I am going to try and load a bigger 3-16 Letter Words.txt file. NOPE!
  42. ' The 3-12 list is loading again today, but no garantee it will work for others or me in future as it quit last night.
  43. ' Try loading up Browser and run other stuff, maybe QB64 variable length string loads in one array is limited. Still OK.
  44. ' Have no idea why it works for .5 day and later doesn't ????
  45.  
  46. ' 2022-01-16 Boggle 2 post: I will have the 3 to 10 Letter Words.txt file to load into game if you or I have problems loading the
  47. ' 3 to 12 Letter Words.txt file. So just Boggle 2 and 2 word files in zip.
  48.  
  49. '  =================================== Sound Signals =========================================================
  50. ' Oh a high pitched peep is a sound signalling that the word enetered can't be legally built from board.
  51. ' A Beep is a signal the word is not in dictionary.
  52. '=============================================================================================================
  53.  
  54. ' 2022-01-16 Boggle 3 this may be a fork but I am going to not load the Collins word file into an array
  55. ' but use an RA file I created with a bas Make code. I made the word list and the words with definitions of all words!
  56. ' BTW the maxLen of a word was 15.
  57. ' So first step is to convert code to this change and see how it does, will RA be too slow? We will save time
  58. ' loading the file into an array but lose it throughout the game?
  59. ' Hey not loading the word file(s) is working fine! AI doesn't seem to take any longer to go through the words.
  60. ' Now you can lookup the words you don't know, press h for help with word after the reckoning score update.
  61.  
  62. Const NTopWord = 279496
  63. Const TimeLimit = 180 ' actual game time is 3 minutes = 180 secs
  64. Const PrintLine = 20 ' row where I can start display of user inputs below Board display on screen.
  65. Dim Shared Board$(3, 3) ' < this is over sized array # words is 279369 try jacks idea of oversized array
  66. Dim Shared As Long f48, f30, dx(7), dy(7)
  67. Dim Shared As Double BoggleTime, elapsed
  68. Dim Shared rec15 As String * 15
  69. Dim w$, k$, player$, AI$
  70. Dim rec237 As String * 237
  71. Dim As Long pScore, aiScore, uba, ubp, i, j, mx, my, mbL, mbR, row, col, ndef
  72. Screen _NewImage(800, 700, 32)
  73. _ScreenMove 200, 40
  74.  
  75.     NewBoard ' display timer and allow input of words from user for 3 minutes
  76.     DisplayBoard
  77.     elapsed = 0: player$ = "": w$ = ""
  78.     While elapsed < TimeLimit ' do stuff   <<<<<<<<<<<<<<<<<<<<<< off while debug wordBuildOK
  79.         Cls
  80.         DisplayBoard
  81.         elapsed = Timer(.01) - BoggleTime
  82.         If elapsed < 0 Then elapsed = 24 * 60 * 60 + Timer(.01) - BoggleTime ' midnight problem add aday of seconds to timer and subtr boogle
  83.         _Font f48
  84.         Color &HFFFFFF00
  85.         _PrintString (300, 240), _Trim$(Str$(TimeLimit - Int(elapsed)))
  86.  
  87.         _Font 16
  88.         Color _RGB32(200, 200, 255)
  89.  
  90.         'some help
  91.         Locate 5, 50: Print "Use Keyboard or Mouse to add letters to word."
  92.         Locate 6, 50: Print " Backspace will clear last letter."
  93.         Locate 7, 50: Print " Esc or Ctrl+C will clear word started."
  94.         Locate 8, 50: Print " Enter or Right Mouse to checks word > list."
  95.  
  96.         'add mouse controls
  97.         While _MouseInput: Wend
  98.         mx = _MouseX: my = _MouseY: mbL = _MouseButton(1): mbR = _MouseButton(2)
  99.         row = (my - 80) / 60: col = (mx - 80) / 60 ' ?? not -60 but more
  100.         If mbL Then
  101.             _Delay .25
  102.             If -1 < row And row < 4 Then
  103.                 If -1 < col And col < 4 Then
  104.                     w$ = w$ + Board$(col, row) ' add letter to w$ build
  105.                 End If
  106.             End If
  107.         End If
  108.         If mbR Then ' same as hitting Enter on keyboard
  109.             _Delay .25
  110.             If wordBuildOK&(w$) Then ' legal build
  111.                 If Find&(w$) Then ' check words in dictionary before add to player$
  112.                     If player$ = "" Then player$ = w$ Else player$ = player$ + " " + w$
  113.                 Else
  114.                     Beep
  115.                 End If
  116.             Else
  117.                 Sound 3000, 1
  118.             End If
  119.             w$ = ""
  120.         End If
  121.  
  122.         k$ = UCase$(InKey$) ' compare to all caps
  123.         If Len(k$) Then 'handle 1 and 2 char key presses, maybe replace with _keyhit later
  124.             Select Case Len(k$)
  125.                 Case 1
  126.                     Select Case Asc(k$)
  127.                         Case 3 'Ctrl + C   another way to clear?
  128.                             w$ = ""
  129.                         Case 8 ' backspace          more to do
  130.                             If Len(w$) Then w$ = Left$(w$, Len(w$) - 1)
  131.                         Case 13
  132.                             If wordBuildOK&(w$) Then ' legal build
  133.                                 If Find&(w$) Then ' check words in dictionary before add to player$
  134.                                     If player$ = "" Then player$ = w$ Else player$ = player$ + " " + w$
  135.                                 Else
  136.                                     Beep
  137.                                 End If
  138.                             Else
  139.                                 Sound 3000, 1
  140.                             End If
  141.                             w$ = ""
  142.                         Case 27 'esc
  143.                             w$ = "" ': exit ?
  144.                         Case Else
  145.                             If 63 < Asc(k$) And Asc(k$) < 91 Then w$ = w$ + k$
  146.                     End Select
  147.             End Select
  148.         End If
  149.         Locate PrintLine, 1: Print w$
  150.         Locate PrintLine + 1, 1: Print player$
  151.         _Display
  152.         _Limit 60
  153.     Wend
  154.     _PrintString (400, 240), "Times up!"
  155.  
  156.     'lets see what the AI comes up with
  157.     AI$ = AIwords$(5) ' new much larger much longer words
  158.     Locate PrintLine + 8, 1: Print "AI: "; AI$
  159.     Print
  160.     Print "          zzz... Press any for the time of reckoning,"
  161.     Print "   matching words on 2 lists will be removed and the round scored."
  162.  
  163.     k$ = InKey$ ' sleep aint working even with _keyclear
  164.     While Len(InKey$) = 0: k$ = InKey$: _Limit 60: Wend: k$ = ""
  165.     ' evalaute results (remove matching words in lists) and score
  166.  
  167.     Cls
  168.     DisplayBoard
  169.     _Font 16
  170.     player$ = removeRepeats$(player$) ' remove words input 2 or more times
  171.     ReDim p(1 To 1) As String 'setup to convert to an array
  172.     Split player$, " ", p() ' convert player$ to array
  173.     ubp = UBound(p) ' quick count of words because array is base 1
  174.  
  175.     ReDim a(1 To 1) As String ' likewise for AI's string list
  176.     Split AI$, " ", a()
  177.     uba = UBound(a)
  178.  
  179.     For i = 1 To uba ' Remove matching words from lists
  180.         For j = 1 To ubp
  181.             If a(i) = p(j) Then a(i) = "": p(j) = ""
  182.         Next
  183.     Next
  184.     pScore = score&(p()): aiScore = score&(a()) ' scores done
  185.  
  186.     AI$ = "": player$ = "" ' reset strings for rebuild
  187.     For i = 1 To uba ' rebuild AI's non matching words
  188.         If a(i) <> "" Then AI$ = AI$ + " " + a(i)
  189.     Next
  190.     For i = 1 To ubp ' rebuild players non matching words (if any)
  191.         If p(i) <> "" Then player$ = player$ + " " + p(i)
  192.     Next
  193.     'display results of reckoning for round
  194.     Locate PrintLine, 1: Print "Player:"; player$; " Score:"; pScore
  195.     Locate PrintLine + 8, 1: Print "AI:"; AI$; " Score:"; aiScore
  196.  
  197.     ' new option to lookup words!!!
  198.     h_again:
  199.     Print: Print "      zzz... read em and weep, press h to lookup word, any for another round "
  200.     k$ = InKey$
  201.     While Len(k$) = 0: k$ = InKey$: _Limit 60: Wend
  202.     If k$ = "h" Then
  203.         Locate PrintLine + 1, 1: Print Space$(300);
  204.         Locate PrintLine + 1, 1
  205.         Input "Enter word to lookup "; w$
  206.         w$ = UCase$(w$)
  207.         ndef = Find&(w$)
  208.         If ndef Then
  209.             Open "Collins Words and Defs.RA" For Random As #2 Len = 237
  210.             Get #2, ndef, rec237
  211.             Close #2
  212.             Print rec237
  213.         Else
  214.             Beep
  215.         End If
  216.         GoTo h_again
  217.     End If
  218.  
  219. Sub NewBoard
  220.     Static BeenHere, Di$(), Numbers()
  221.     Dim As Long i, r, c
  222.  
  223.     If BeenHere = 0 Then 'load and initialize all the one time stuff
  224.         ' load fonts
  225.         f48 = _LoadFont("Arial.ttf", 48, "MONOSPACE")
  226.         f30 = _LoadFont("Arial.ttf", 30, "MONOSPACE")
  227.         If f48 <= 0 Then Print "Sub NewBoard: Font did not load, goodbye.": End
  228.  
  229.         'we call on this later
  230.         Open "Collins_Word_List.RA" For Random As #1 Len = 15
  231.  
  232.         ' load dx(), dy() for testing the legality of words built from board
  233.         dx(0) = -1: dy(0) = -1 ' this is for AI to find words
  234.         dx(1) = 0: dy(1) = -1
  235.         dx(2) = 1: dy(2) = -1
  236.         dx(3) = -1: dy(3) = 0
  237.         dx(4) = 1: dy(4) = 0
  238.         dx(5) = -1: dy(5) = 1
  239.         dx(6) = 0: dy(6) = 1
  240.         dx(7) = 1: dy(7) = 1
  241.  
  242.         ' These are the 17 Dice with 6 Faces of a Letter need for Boggle
  243.         Dim Di$(0 To 16)
  244.         Di$(0) = "PACEMD"
  245.         Di$(1) = "RIFOBX"
  246.         Di$(2) = "IFEHEY"
  247.         Di$(3) = "DENOWS"
  248.         Di$(4) = "UTOKND"
  249.         Di$(5) = "HMSRAO"
  250.         Di$(6) = "LUPETS"
  251.         Di$(7) = "ACITOA"
  252.         Di$(8) = "YLGKUE"
  253.         Di$(9) = "QBMJOA"
  254.         Di$(10) = "EHISPN"
  255.         Di$(11) = "VETIGN"
  256.         Di$(12) = "BALIYT"
  257.         Di$(13) = "EZAVND"
  258.         Di$(14) = "RALESC"
  259.         Di$(15) = "UWILRG"
  260.         Di$(16) = "AEIOUU" ' b+ mod Boggle 2 to remove all special handling of Q words!!!
  261.  
  262.         Dim Numbers(0 To 16) ' load numbers for shuffling die order
  263.         For i = 0 To 16
  264.             Numbers(i) = i
  265.         Next
  266.         BeenHere = -1
  267.     End If
  268.  
  269.     'now get the game going
  270.     For i = 16 To 1 Step -1 'shuffle die
  271.         Swap Numbers(i), Numbers(Int(Rnd * (i + 1)))
  272.     Next
  273.     'For i = 1 To 16: Print Numbers(i);: Next: Print   ' check the shuffle
  274.     For i = 0 To 15 'choosing random face of die = 1 Letter
  275.         Index2ColRow i, c, r
  276.         Board$(c, r) = Mid$(Di$(Numbers(i)), Int(Rnd * 6) + 1, 1) ' one die gets is left out now Boggle 2
  277.     Next
  278.     ' now set timer + 180
  279.     BoggleTime = Timer(.01)
  280.     _Font 16
  281.  
  282. Sub DisplayBoard
  283.     Dim row, col
  284.     _Font f48
  285.     For row = 0 To 3 '  display the board
  286.         For col = 0 To 3
  287.             Line ((col + 1) * 60 - 5, (row + 1) * 60 - 5)-Step(54, 54), &HFF2020FF, BF 'face color or die
  288.             Color &HFF661111 'shade
  289.             _PrintString ((col + 1) * 60 + 2, (row + 1) * 60 + 2), Board$(col, row)
  290.             Color &HFFBBBBBB 'letter
  291.             _PrintString ((col + 1) * 60, (row + 1) * 60), Board$(col, row)
  292.         Next
  293.     Next
  294.  
  295. 'might not need in final cut
  296. Function ColRow2Index& (row As Long, col As Long) ' convert a board letter to index (not needed yet?)
  297.     ColRow2Index& = row * 4 + col
  298.  
  299. Sub Index2ColRow (indexIn As Long, rowOut As Long, colOut As Long) 'convert die index to board col, row
  300.     colOut = indexIn Mod 4: rowOut = indexIn \ 4
  301.  
  302. Sub Split (SplitMeString As String, delim As String, loadMeArray() As String)
  303.     Dim curpos As Long, arrpos As Long, LD As Long, dpos As Long 'fix use the Lbound the array already has
  304.     curpos = 1: arrpos = LBound(loadMeArray): LD = Len(delim)
  305.     dpos = InStr(curpos, SplitMeString, delim)
  306.     Do Until dpos = 0
  307.         loadMeArray(arrpos) = Mid$(SplitMeString, curpos, dpos - curpos)
  308.         arrpos = arrpos + 1
  309.         If arrpos > UBound(loadMeArray) Then ReDim _Preserve loadMeArray(LBound(loadMeArray) To UBound(loadMeArray) + 1000) As String
  310.         curpos = dpos + LD
  311.         dpos = InStr(curpos, SplitMeString, delim)
  312.     Loop
  313.     loadMeArray(arrpos) = Mid$(SplitMeString, curpos)
  314.     ReDim _Preserve loadMeArray(LBound(loadMeArray) To arrpos) As String 'get the ubound correct
  315.  
  316. Function Find& (x$) ' if I am using this only to find words in dictionary, I can mod to optimize
  317.     ' the RA file is opened and ready for gets
  318.     Dim As Long low, hi, test
  319.     Dim w$
  320.     If Len(x$) < 3 Then Exit Function ' words need to be 3 letters
  321.     low = 1: hi = NTopWord
  322.     While low <= hi
  323.         test = Int((low + hi) / 2)
  324.         Get #1, test, rec15
  325.         w$ = _Trim$(rec15)
  326.         If w$ = x$ Then
  327.             Find& = test: Exit Function
  328.         Else
  329.             If w$ < x$ Then low = test + 1 Else hi = test - 1
  330.         End If
  331.     Wend
  332.  
  333. ' This function checks to see that the word w$ is legally constructable with the given board.
  334. ' This function requires the recurive Function findCell& (startX As Long, startY As Long, word$, index As Long, Arr$())
  335. Function wordBuildOK& (w$)
  336.     Dim As Long r, c, test
  337.     Dim copy$(-1 To 4, -1 To 4), first$
  338.     If Len(w$) < 3 Then Exit Function ' words need to be 3 letters
  339.     For r = 0 To 3
  340.         For c = 0 To 3
  341.             copy$(c, r) = Board$(c, r)
  342.         Next
  343.     Next
  344.     first$ = Mid$(w$, 1, 1)
  345.     For r = 0 To 3
  346.         For c = 0 To 3
  347.             If copy$(c, r) = first$ Then 'cell letter matches first letter in word
  348.                 test = findCell&(c, r, w$, 2, copy$())
  349.                 If test Then wordBuildOK& = -1: Exit Function ' ah ha! maybe it keeps trying when we are supposed to be done, fix?
  350.             End If
  351.         Next
  352.     Next
  353.  
  354. 'recursively called starting from wordBuildOK&
  355. Function findCell& (startX As Long, startY As Long, word$, index As Long, Arr$()) ' want to setup recursive searcher
  356.     Dim As Long d, x, y, i, r, c, test
  357.     Dim w$
  358.     'make own set of variables for this function  (attempt to debug but did not fix anything)
  359.     Dim a$(-1 To 4, -1 To 4)
  360.     For r = 0 To 3
  361.         For c = 0 To 3
  362.             a$(c, r) = Arr$(c, r)
  363.         Next
  364.     Next
  365.     i = index: w$ = word$: y = startY: x = startX
  366.     If i > Len(w$) Then findCell = -1: Exit Function
  367.     a$(x, y) = "" 'so wont be used again
  368.     For d = 0 To 7
  369.         If a$(x + dx(d), y + dy(d)) = Mid$(w$, i, 1) Then
  370.             test = findCell&(x + dx(d), y + dy(d), w$, i + 1, a$())
  371.             If test Then findCell& = -1: Exit Function
  372.         End If
  373.     Next
  374.  
  375. Function AIwords$ (timeLimit As Double) 'returns a space delimiter string of 1 point words that can be constructed from board in limited time
  376.     Dim As Double startTime, checkTime
  377.     Dim As Long i, r, c, OK, dp, l
  378.     Dim l$, letters$, b$, wd$
  379.     startTime = Timer(.01)
  380.     'ub = UBound(WordList$)  now is NTopWord
  381.     ' get a non redundant list of letters from board and put them in alpha order
  382.     For r = 0 To 3
  383.         For c = 0 To 3
  384.             l$ = Board$(c, r)
  385.             If (r = 0) And (c = 0) Then
  386.                 letters$ = l$
  387.             Else
  388.                 If InStr(letters$, l$) <= 0 Then '  insrt letter
  389.                     OK = 0
  390.                     For i = 1 To Len(letters$) '            where?
  391.                         If Asc(l$) < Asc(letters$, i) Then ' here!
  392.                             letters$ = Mid$(letters$, 1, i - 1) + l$ + Mid$(letters$, i)
  393.                             OK = -1: Exit For
  394.                         End If
  395.                     Next
  396.                     If OK = 0 Then letters$ = letters$ + l$
  397.                 End If
  398.             End If
  399.         Next
  400.     Next
  401.     'check if this is OK so far  OK finally!  This is 3rd time I needed to exit when found
  402.     ' AIwords$ = letters$
  403.     'now letters of board are in alpha order
  404.     dp = 1 'place in dict
  405.     For l = 1 To Len(letters$) ' advance place in list$ by one until the word > letter
  406.         Get #1, dp, rec15
  407.         wd$ = _Trim$(rec15)
  408.         While Asc(wd$, 1) < Asc(letters$, l)
  409.             dp = dp + 1
  410.             If dp > NTopWord Then GoTo fini
  411.             If Timer(.01) - startTime < 0 Then checkTime = Timer(.01) + 24 * 60 * 60 Else checkTime = Timer(.01)
  412.             If checkTime - startTime > timeLimit Then GoTo fini
  413.             Get #1, dp, rec15
  414.             wd$ = _Trim$(rec15)
  415.         Wend
  416.         'now start testing words
  417.         While Asc(wd$, 1) = Asc(letters$, l)
  418.             If wordBuildOK&(wd$) Then
  419.                 If b$ = "" Then b$ = wd$ Else b$ = b$ + " " + wd$
  420.             End If
  421.             dp = dp + 1
  422.             If dp > NTopWord Then GoTo fini
  423.             If Timer(.01) - startTime < 0 Then checkTime = Timer(.01) + 24 * 60 * 60 Else checkTime = Timer(.01)
  424.             If checkTime - startTime > timeLimit Then GoTo fini
  425.             Get #1, dp, rec15
  426.             wd$ = _Trim$(rec15)
  427.         Wend
  428.     Next
  429.  
  430.     fini:
  431.     AIwords$ = b$
  432.  
  433. Function removeRepeats$ (s$) ' s$ is space delimited word list
  434.     ReDim t$(1 To 1), b$
  435.     Dim As Long ub, i, j, ok
  436.     Split s$, " ", t$()
  437.     ub = UBound(t$)
  438.     For i = 1 To ub
  439.         ok = -1
  440.         For j = 1 To i - 1
  441.             If t$(i) = t$(j) Then ok = 0: Exit For
  442.         Next
  443.         If ok Then
  444.             If b$ = "" Then b$ = t$(i) Else b$ = b$ + " " + t$(i)
  445.         End If
  446.     Next
  447.     removeRepeats$ = b$
  448.  
  449. Function score& (a() As String)
  450.     Dim As Long i, s
  451.     For i = 1 To UBound(a)
  452.         Select Case Len(a(i))
  453.             Case 3, 4: s = s + 1
  454.             Case 5: s = s + 2
  455.             Case 6: s = s + 3
  456.             Case 7: s = s + 5
  457.             Case Is > 7: s = s + 11
  458.         End Select
  459.     Next
  460.     score& = s
  461.  

Boggle 3 works zip has the 2 Collins converted files along with the Boggle 3.bas source.
Title: Re: Boggle play against AI - WIP
Post by: jack on January 16, 2022, 10:13:06 pm
the AI is not following the rules
Quote
Each player searches for words that can be constructed from the letters of sequentially adjacent cubes, where "adjacent" cubes are those horizontally, vertically, and diagonally neighboring.
looks like I need to practice, here's one online https://wordshake.com/boggle
Title: Re: Boggle play against AI - WIP
Post by: _vince on January 17, 2022, 12:32:58 am
Boggle 2 update with Qwords special handling gone

BONED

Boggle 2 Before Reckoning:

ANAL

Solved the problem with loading all the words.

DIARREA!



Title: Re: Boggle play against AI - WIP
Post by: bplus on January 17, 2022, 08:36:17 am
the AI is not following the rules looks like I need to practice, here's one online https://wordshake.com/boggle

Looks like they follow different rules for scoring, I used Wiki see first post but then I got rid of special handling of Q and   added another die to have more U's around.
Title: Re: Boggle play against AI - WIP
Post by: bplus on January 17, 2022, 08:38:19 am
BONED

ANAL

DIARREA!

Not your cup of tea I take it.
Title: Re: Boggle play against AI - WIP
Post by: bplus on January 17, 2022, 11:42:25 am
BONED

ANAL

DIARREA!

Internet lookup:
Quote
Did you mean: DIARRHEA

@_vince
Update: ADOLESCEN... oh hey maybe you think you can use any letter from the board at any time?
Nope, each letter must neighbor another (8 for middle square) AND no going back to a used letter for Boggle.

This is why I was so happy I could detect in code a legal build from the board. :)
Title: Re: Boggle play against AI - WIP
Post by: bplus on January 21, 2022, 01:08:12 pm
Coming soon, Boggle Trainer:
1. Return the Q(u) die and u Insert function, remove 17th die so normal game again.
2. Keep words and points for Player, AI being perfect don't need to show.
3, Filter out all the words with () in definition that majority of people never use.
4. Fun surprise kids might like, I know I do :)

If you are playing another human in a real game and they try one of those () words, I would insist they must be able to define the word, otherwise they are just making up stuff, testing different permutations.
Title: Re: Boggle play against AI - WIP
Post by: bplus on January 21, 2022, 10:30:33 pm
I think I've got all the bugs and cleaned up:
Code: QB64: [Select]
  1. _Title "Boggle Trainer" ' b+ start 2022-01-12 2022-01-21 use this code as Boggle Trainer AI is too good.
  2. ' It is possible to construct a word the Collins Dictionary has and AI wont show because I am not listing
  3. ' exotic Dictionary words that come from Hawiaan, Shakespear, Scots, Hindi, South African, French, Latin, Tagalog,
  4. ' Spencer, obsolete, Dutch, Welsh, archaic, Greek,...   all enclosed in ()'s and used as filter.
  5.  
  6. ' Scabble Word List and Dictionary
  7. ' ref dictionary: https://boardgames.stackexchange.com/questions/38366/latest-collins-scrabble-words-list-in-text-file
  8. ' Die configurations
  9. ' https://boardgames.stackexchange.com/questions/29264/boggle-what-is-the-dice-configuration-for-boggle-in-various-languages
  10. ' pointing finger image
  11. ' http://clipart-library.com/pointing-finger-images.html
  12. ' Thank you!
  13.  
  14. ' 2022-01-14 status have the board display separated out from New Board and Game initialization
  15. ' have timer system and display working next to board display
  16. ' have player input system working
  17. '   1. check word in dictionary working
  18. '   2. check that word can legally be built from board is funky! sometimes works sometimes not 50/50 for legit builds yuck
  19. '     How to debug? Todays task is get this abolutely critical function working so the AI would be fairly easy to do.
  20. ' Oh I had to modify the Collins Word list to get a size that would fit into a variable length string array, 279496 words
  21. ' kept crashing QB64 without error messages when reached about 260,000 words so now I have a list for words with lengths of
  22. ' between 3 and 10, still 199,651 words!
  23.  
  24. ' 2202-01-14 Hurray! Function wordBuildOK& is fixed, a very simple logic bug. I continued to check for builds of word after
  25. ' I got confirmation one of the builds was possible. It kept checking if there was more than one place the first letter of word
  26. ' appeared. The function is crucial for AI testing if words are buildable. I can get started on that today now that bug is
  27. ' fixed. Man! I had this fixed almost immediately, I had to exit the Function when the recursive function findCell& called from
  28. ' it found a positive result.  1/21 Yes this game rides on this function!
  29.  
  30. ' 2022-01-15 qw$() function to handle the Q letter, function to handle scoring, main removes matching words on the 2 lists.
  31. ' you should be able to type 2 letters with (qu) square and get a 3 letter word.  1/21 This was fine fix!
  32.  
  33. ' 2022-01-15 screw the die with face Q(u) with implied u, adding a 17th die with AEIOUU and removing all the special handling
  34. ' for that implied u crap! I don't think this will ruin the game but only make it better and less difficlut to code.
  35. ' Fix the spelling of reckoning :)  OK new word files made for Boggle 2: AI Check Words is HUGE! Almost 35 X's more and longer
  36. ' words, words 5 letters and longer. The Word List file has been expanded from 3 to 10 letter words to 3 to 12 letter words.
  37. ' Dang! It's worse the points the AI collects from big words times the point values they are worth...
  38. ' OK more handicap, AI concedes all 3-6 letters words to player and gets points only for 7+ letter words. Is it fair now?
  39. ' Too much! Try concedes 3-5 but fight for 6+ letter words.  1/21 Abandoned 17th die and fixed handling of Q-words.
  40.  
  41. ' 2022-01-16 Yesterday I was able to use 2 large word lists from : 3 - 12 Letter Words.txt (249,702)
  42. ' and AI Check Words.txt (229,745) they loaded fine all afternoon. Then I decided to use the 3-12 letter words for the
  43. ' AI check words list so I only had to load one file! So I change the code to do that and suddenly the 3-12 Letter Words.txt
  44. ' was quitting like there were more words than it could load (no error message, no subscript out of range you get with an array
  45. ' dim'd too small for # of items to put in it), QB64 just bugged out and quit ??? Just like when I was trying to load the
  46. ' whole file.   I don't like this shakey file situation, found much better method see Boggle 3.
  47.  
  48. ' 2022-01-16 Boggle 2 post: I will have the 3 to 10 Letter Words.txt file to load into game if you or I have problems loading the
  49. ' 3 to 12 Letter Words.txt file. So just Boggle 2 and 2 word files in zip. Abandoned, one RA file now 1/21.
  50.  
  51. ' =================================== Sound Signals =========================================================
  52. ' A high pitched squeak is a sound signalling that the word entered can't be legally built from board.
  53. ' A Beep is a signal the word is not in dictionary.
  54. '=============================================================================================================
  55.  
  56. ' 2022-01-16 Boggle 3 this may be a fork but I am going to not load the Collins word file into an array
  57. ' but use an RA file I created with a bas Make code. I made the word list and the words with definitions of all words!
  58. ' BTW the maxLen of a word was 15.
  59. ' So first step is to convert code to this change and see how it does, will RA be too slow? We will save time
  60. ' loading the file into an array but lose it throughout the game?
  61. ' Hey not loading the word file(s) is working fine! AI doesn't seem to take any longer to go through the words.
  62. ' Now you can lookup the words you don't know, press h for help with word after the reckoning score update.
  63. ' This is not a fork but a Keeper! Love the word lookup!
  64.  
  65. ' 2022-01-17 Boggle 4, debating restore of special handling of Q words, adding a pointing finger for mouse pointer
  66. ' 2022-01-21 Yes make this app a Boggle Trainer
  67. '  1. Return the Q(u) die and u Insert function, remove 17th die so normal game again. OK done
  68. '  2. Keep words and points for Player, AI being perfect don't need to show. Just relabel points as possible.
  69. '  3, Filter out all the words with () in definition that majority of people never use. Done
  70. '  4. Fun surprise kids might like, I know I do :)  Done
  71.  
  72. Const NTopWord = 279496
  73. Const TimeLimit = 180 ' actual game time is 3 minutes = 180 secs
  74. Const PrintLine = 20 ' row where I can start display of user inputs below Board display on screen.
  75. Dim Shared Board$(3, 3)
  76. Dim Shared As Long f48, f30, dx(7), dy(7)
  77. Dim Shared As Double BoggleTime, elapsed
  78. Dim Shared rec237 As String * 237 ' dictionary definitions lookup
  79. Dim Shared rec15 As String * 15 ' dictionary words lookup
  80.  
  81. 'Main Locals
  82. Dim w$, k$, player$, AI$, df$
  83. Dim As Long pScore, aiScore, mx, my, mbL, mbR, row, col, pointer
  84.  
  85. Screen _NewImage(800, 600, 32)
  86. _ScreenMove 200, 40
  87. pointer = _LoadImage("point.png") ' load the "handy" pointer ha, ha
  88. _ClearColor &HFF000000, pointer
  89.  
  90.     Cls
  91.     NewBoard
  92.     DisplayBoard
  93.     ' display timer and allow input of words from user for 3 minutes
  94.     elapsed = 0: player$ = "": w$ = ""
  95.     While elapsed < TimeLimit ' do stuff   <<<<<<<<<<<<<<<<<<<<<< off while debug wordBuildOK
  96.         Cls
  97.         DisplayBoard
  98.         elapsed = Timer(.01) - BoggleTime
  99.         If elapsed < 0 Then elapsed = 24 * 60 * 60 + Timer(.01) - BoggleTime ' midnight problem add aday of seconds to timer and subtr boogle
  100.         _Font f48
  101.         Color &HFFFFFF00
  102.         _PrintString (300, 240), _Trim$(Str$(TimeLimit - Int(elapsed)))
  103.  
  104.         _Font 16 ' some help instructions
  105.         Color _RGB32(200, 200, 255)
  106.         Locate 5, 50: Print "Use Keyboard or Mouse to add letters to word."
  107.         Locate 6, 50: Print " Backspace will clear last letter."
  108.         Locate 7, 50: Print " Esc or Ctrl+C will clear word started."
  109.         Locate 8, 50: Print " Enter or Right Mouse to checks word > list."
  110.  
  111.         'add mouse controls with image pointer
  112.         While _MouseInput: Wend
  113.         mx = _MouseX: my = _MouseY: mbL = _MouseButton(1): mbR = _MouseButton(2)
  114.         _PutImage (mx - 45, my - 14)-Step(225, 225), pointer, 0
  115.         row = (my - 80) / 60: col = (mx - 80) / 60 ' ?? not -60 but more
  116.         If mbL Then
  117.             _Delay .25
  118.             If -1 < row And row < 4 Then
  119.                 If -1 < col And col < 4 Then
  120.                     w$ = w$ + Board$(col, row) ' add letter to w$ build
  121.                 End If
  122.             End If
  123.         End If
  124.         If mbR Then ' same as hitting Enter on keyboard
  125.             _Delay .25
  126.             If wordBuildOK&(w$) Then ' legal build
  127.  
  128.                 ' check words with or without U insertion after Q in word check word wo insertion first! so can get qat
  129.                 If Find&(w$) Then ' check words in dictionary before add to player$
  130.                     If player$ = "" Then player$ = w$ Else player$ = player$ + " " + w$
  131.                 ElseIf Find&(qw$(w$)) Then ' check words in dictionary before add to player$
  132.                     If player$ = "" Then player$ = qw$(w$) Else player$ = player$ + " " + qw$(w$)
  133.                 Else
  134.                     Beep
  135.                 End If
  136.             Else
  137.                 Sound 3000, 1
  138.             End If
  139.             w$ = ""
  140.         End If
  141.  
  142.         k$ = UCase$(InKey$) ' compare to all caps
  143.         If Len(k$) Then 'handle 1 and 2 char key presses, maybe replace with _keyhit later
  144.             Select Case Len(k$)
  145.                 Case 1
  146.                     Select Case Asc(k$)
  147.                         Case 3 'Ctrl + C   another way to clear?
  148.                             w$ = ""
  149.                         Case 8 ' backspace          more to do
  150.                             If Len(w$) Then w$ = Left$(w$, Len(w$) - 1)
  151.                         Case 13
  152.                             If wordBuildOK&(w$) Then ' legal build
  153.  
  154.                                 ' check words with or without U insertion after Q in word
  155.                                 If Find&(w$) Then ' check words in dictionary before add to player$
  156.                                     If player$ = "" Then player$ = w$ Else player$ = player$ + " " + w$
  157.                                 ElseIf Find&(qw$(w$)) Then ' check words in dictionary before add to player$
  158.                                     If player$ = "" Then player$ = qw$(w$) Else player$ = player$ + " " + qw$(w$)
  159.                                 Else
  160.                                     Beep
  161.                                 End If
  162.                             Else
  163.                                 Sound 3000, 1
  164.                             End If
  165.                             w$ = ""
  166.                         Case 27 'esc
  167.                             w$ = "" ': exit ?
  168.                         Case Else
  169.                             If 63 < Asc(k$) And Asc(k$) < 91 Then w$ = w$ + k$
  170.                     End Select
  171.             End Select
  172.         End If
  173.         Locate PrintLine, 1: Print w$
  174.         Locate PrintLine + 1, 1: Print player$
  175.         _Display
  176.         _Limit 60
  177.     Wend
  178.  
  179.     _AutoDisplay ' reproduce the screen without the hand image
  180.     _MouseShow ' this is handy for tracking words AI finds
  181.     Cls
  182.     DisplayBoard
  183.     elapsed = Timer(.01) - BoggleTime
  184.     If elapsed < 0 Then elapsed = 24 * 60 * 60 + Timer(.01) - BoggleTime ' midnight problem add aday of seconds to timer and subtr boogle
  185.     _Font f48
  186.     Color &HFFFFFF00
  187.     _PrintString (300, 240), _Trim$(Str$(TimeLimit - Int(elapsed)))
  188.     _Font f30
  189.     _PrintString (400, 240), "Times up!"
  190.     _Font 16
  191.     Color _RGB32(200, 200, 255)
  192.     Locate PrintLine + 1, 1: Print player$
  193.  
  194.     'lets see what the AI comes up with
  195.     AI$ = AIwords$(10) ' new much larger much longer words
  196.     Locate PrintLine + 8, 1: Print "AI: "; AI$
  197.     Print: Print "          zzz... Press any for the time of reckoning,"
  198.  
  199.     _KeyClear
  200.     k$ = ""
  201.     k$ = InKey$ ' sleep aint working even with _keyclear
  202.     While Len(InKey$) = 0: k$ = InKey$: _Limit 60: Wend: k$ = ""
  203.  
  204.     Cls
  205.     DisplayBoard
  206.     _Font 16
  207.     player$ = removeRepeats$(player$) ' remove words input 2 or more times
  208.     ReDim p(1 To 1) As String ' setup to score words convert to an array
  209.     Split player$, " ", p() ' convert player$ to array
  210.  
  211.     ReDim a(1 To 1) As String ' likewise for AI's string list
  212.     Split AI$, " ", a()
  213.     pScore = score&(p()): aiScore = score&(a()) ' scores done
  214.  
  215.     'display results of reckoning for round
  216.     Locate PrintLine, 1: Print "Player:";
  217.     Locate PrintLine, 8: Print player$; " Score:"; pScore
  218.     Locate PrintLine + 8, 1: Print "AI:";
  219.     Locate PrintLine + 8, 3: Print AI$; " Common Words Available Score:"; aiScore
  220.  
  221.     ' new option to lookup words!!!
  222.     h_again:
  223.     Locate PrintLine + 5, 1: Print "      zzz... press h to lookup word, any for another round "
  224.     _KeyClear
  225.     k$ = ""
  226.     k$ = InKey$
  227.     While Len(k$) = 0: k$ = InKey$: _Limit 60: Wend
  228.     If k$ = "h" Then
  229.         Locate PrintLine + 1, 1: Print Space$(300);
  230.         Locate PrintLine + 1, 1
  231.         Input "Enter word to lookup "; w$
  232.         df$ = defineWord$(w$)
  233.         If df$ <> "" Then Print df$ Else Beep 'not found
  234.         GoTo h_again
  235.     End If
  236.  
  237. Sub NewBoard
  238.     Static BeenHere, Di$(), Numbers()
  239.     Dim As Long i, r, c
  240.  
  241.     If BeenHere = 0 Then 'load and initialize all the one time stuff
  242.         ' load fonts
  243.         f48 = _LoadFont("Arial.ttf", 48, "MONOSPACE")
  244.         f30 = _LoadFont("Arial.ttf", 30, "MONOSPACE")
  245.         If f48 <= 0 Then Print "Sub NewBoard: Font did not load, goodbye.": End
  246.  
  247.         ' we call on this later
  248.         Open "Collins_Word_List.RA" For Random As #1 Len = 15
  249.  
  250.         ' load dx(), dy() for testing the legality of words built from board
  251.         dx(0) = -1: dy(0) = -1 ' this is for AI to find words
  252.         dx(1) = 0: dy(1) = -1
  253.         dx(2) = 1: dy(2) = -1
  254.         dx(3) = -1: dy(3) = 0
  255.         dx(4) = 1: dy(4) = 0
  256.         dx(5) = -1: dy(5) = 1
  257.         dx(6) = 0: dy(6) = 1
  258.         dx(7) = 1: dy(7) = 1
  259.  
  260.         ' These are the 16 Dice with 6 Faces of a Letter need for Boggle back to orig version
  261.         Dim Di$(0 To 15)
  262.         Di$(0) = "PACEMD"
  263.         Di$(1) = "RIFOBX"
  264.         Di$(2) = "IFEHEY"
  265.         Di$(3) = "DENOWS"
  266.         Di$(4) = "UTOKND"
  267.         Di$(5) = "HMSRAO"
  268.         Di$(6) = "LUPETS"
  269.         Di$(7) = "ACITOA"
  270.         Di$(8) = "YLGKUE"
  271.         Di$(9) = "QBMJOA"
  272.         Di$(10) = "EHISPN"
  273.         Di$(11) = "VETIGN"
  274.         Di$(12) = "BALIYT"
  275.         Di$(13) = "EZAVND"
  276.         Di$(14) = "RALESC"
  277.         Di$(15) = "UWILRG"
  278.         Dim Numbers(0 To 15) ' load numbers for shuffling die order
  279.         For i = 0 To 15
  280.             Numbers(i) = i
  281.         Next
  282.         BeenHere = -1
  283.     End If
  284.  
  285.     'now get the game going
  286.     For i = 15 To 1 Step -1 'shuffle die
  287.         Swap Numbers(i), Numbers(Int(Rnd * (i + 1)))
  288.     Next
  289.     For i = 0 To 15 'choosing random face of die = 1 Letter
  290.         Index2ColRow i, c, r
  291.         Board$(c, r) = Mid$(Di$(Numbers(i)), Int(Rnd * 6) + 1, 1)
  292.     Next
  293.     BoggleTime = Timer(.01) ' now set timer + 180
  294.     _Font 16
  295.  
  296. Sub DisplayBoard
  297.     Dim row, col
  298.     _Font f48
  299.     For row = 0 To 3 '  display the board
  300.         For col = 0 To 3
  301.             Line ((col + 1) * 60 - 5, (row + 1) * 60 - 5)-Step(54, 54), &HFF2020FF, BF 'face color or die
  302.             If Board$(col, row) = "Q" Then 'If face has a Q it is supposed to be "Qu"
  303.                 _Font f30
  304.                 Color &HFF661111 'shade
  305.                 _PrintString ((col + 1) * 60 - 4, (row + 1) * 60 + 11), "Q"
  306.                 _PrintString ((col + 1) * 60 + 24, (row + 1) * 60 + 11), "U"
  307.                 Color &HFFBBBBBB 'letter
  308.                 _PrintString ((col + 1) * 60 - 7, (row + 1) * 60 + 9), "Q"
  309.                 _PrintString ((col + 1) * 60 + 22, (row + 1) * 60 + 9), "U"
  310.                 _Font f48
  311.             Else
  312.                 Color &HFF661111 'shade
  313.                 _PrintString ((col + 1) * 60 + 2, (row + 1) * 60 + 2), Board$(col, row)
  314.                 Color &HFFBBBBBB 'letter
  315.                 _PrintString ((col + 1) * 60, (row + 1) * 60), Board$(col, row)
  316.             End If
  317.         Next
  318.     Next
  319.  
  320. Sub Index2ColRow (indexIn As Long, rowOut As Long, colOut As Long) 'convert die index to board col, row
  321.     colOut = indexIn Mod 4: rowOut = indexIn \ 4
  322.  
  323. Sub Split (SplitMeString As String, delim As String, loadMeArray() As String)
  324.     Dim curpos As Long, arrpos As Long, LD As Long, dpos As Long 'fix use the Lbound the array already has
  325.     curpos = 1: arrpos = LBound(loadMeArray): LD = Len(delim)
  326.     dpos = InStr(curpos, SplitMeString, delim)
  327.     Do Until dpos = 0
  328.         loadMeArray(arrpos) = Mid$(SplitMeString, curpos, dpos - curpos)
  329.         arrpos = arrpos + 1
  330.         If arrpos > UBound(loadMeArray) Then ReDim _Preserve loadMeArray(LBound(loadMeArray) To UBound(loadMeArray) + 1000) As String
  331.         curpos = dpos + LD
  332.         dpos = InStr(curpos, SplitMeString, delim)
  333.     Loop
  334.     loadMeArray(arrpos) = Mid$(SplitMeString, curpos)
  335.     ReDim _Preserve loadMeArray(LBound(loadMeArray) To arrpos) As String 'get the ubound correct
  336.  
  337. Function Find& (x$) ' if I am using this only to find words in dictionary, I can mod to optimize
  338.     ' the RA file is opened and ready for gets
  339.     Dim As Long low, hi, test
  340.     Dim w$
  341.     If Len(x$) < 3 Then Exit Function ' words need to be 3 letters
  342.     low = 1: hi = NTopWord
  343.     While low <= hi
  344.         test = Int((low + hi) / 2)
  345.         Get #1, test, rec15
  346.         w$ = _Trim$(rec15)
  347.         If w$ = x$ Then
  348.             Find& = test: Exit Function
  349.         Else
  350.             If w$ < x$ Then low = test + 1 Else hi = test - 1
  351.         End If
  352.     Wend
  353.  
  354. ' This function checks to see that the word w$ is legally constructable with the given board.
  355. ' This function requires the recurive Function findCell& (startX As Long, startY As Long, word$, index As Long, Arr$())
  356. Function wordBuildOK& (w$)
  357.     Dim As Long r, c, test
  358.     Dim copy$(-1 To 4, -1 To 4), first$
  359.     If Len(w$) < 3 Then Exit Function ' words need to be 3 letters
  360.     For r = 0 To 3
  361.         For c = 0 To 3
  362.             copy$(c, r) = Board$(c, r)
  363.         Next
  364.     Next
  365.     first$ = Mid$(w$, 1, 1)
  366.     For r = 0 To 3
  367.         For c = 0 To 3
  368.             If copy$(c, r) = first$ Then 'cell letter matches first letter in word
  369.                 test = findCell&(c, r, w$, 2, copy$())
  370.                 If test Then wordBuildOK& = -1: Exit Function ' ah ha! maybe it keeps trying when we are supposed to be done, fix?
  371.             End If
  372.         Next
  373.     Next
  374.  
  375. 'recursively called starting from wordBuildOK&
  376. Function findCell& (startX As Long, startY As Long, word$, index As Long, Arr$()) ' want to setup recursive searcher
  377.     Dim As Long d, x, y, i, r, c, test
  378.     Dim w$
  379.     'make own set of variables for this function  (attempt to debug but did not fix anything)
  380.     Dim a$(-1 To 4, -1 To 4)
  381.     For r = 0 To 3
  382.         For c = 0 To 3
  383.             a$(c, r) = Arr$(c, r)
  384.         Next
  385.     Next
  386.     i = index: w$ = word$: y = startY: x = startX
  387.     If i > Len(w$) Then findCell = -1: Exit Function
  388.     a$(x, y) = "" 'so wont be used again
  389.     For d = 0 To 7
  390.         If a$(x + dx(d), y + dy(d)) = Mid$(w$, i, 1) Then
  391.             test = findCell&(x + dx(d), y + dy(d), w$, i + 1, a$())
  392.             If test Then findCell& = -1: Exit Function
  393.         End If
  394.     Next
  395.  
  396. Function AIwords$ (timeLimit As Double) 'returns a space delimiter string of 1 point words that can be constructed from board in limited time
  397.     Dim As Double startTime, checkTime
  398.     Dim As Long i, r, c, OK, dp, l
  399.     Dim l$, letters$, b$, wd$
  400.     startTime = Timer(.01)
  401.     ' get a non redundant list of letters from board and put them in alpha order
  402.     For r = 0 To 3
  403.         For c = 0 To 3
  404.             l$ = Board$(c, r)
  405.             If (r = 0) And (c = 0) Then
  406.                 letters$ = l$
  407.             Else
  408.                 If InStr(letters$, l$) <= 0 Then '  insrt letter
  409.                     OK = 0
  410.                     For i = 1 To Len(letters$) '            where?
  411.                         If Asc(l$) < Asc(letters$, i) Then ' here!
  412.                             letters$ = Mid$(letters$, 1, i - 1) + l$ + Mid$(letters$, i)
  413.                             OK = -1: Exit For
  414.                         End If
  415.                     Next
  416.                     If OK = 0 Then letters$ = letters$ + l$
  417.                 End If
  418.             End If
  419.         Next
  420.     Next
  421.     'now letters of board are in alpha order
  422.     dp = 1 'place in dict
  423.     For l = 1 To Len(letters$) ' advance place in list$ by one until the word > letter
  424.         Get #1, dp, rec15
  425.         wd$ = _Trim$(rec15)
  426.         While Asc(wd$, 1) < Asc(letters$, l)
  427.             dp = dp + 1
  428.             If dp > NTopWord Then GoTo fini
  429.             If Timer(.01) - startTime < 0 Then checkTime = Timer(.01) + 24 * 60 * 60 Else checkTime = Timer(.01)
  430.             If checkTime - startTime > timeLimit Then GoTo fini
  431.             Get #1, dp, rec15
  432.             wd$ = _Trim$(rec15)
  433.         Wend
  434.         'now start testing words
  435.         While Asc(wd$, 1) = Asc(letters$, l)
  436.             If wordBuildOK&(wd$) Then
  437.                 If InStr(defineWord$(wd$), "(") <= 0 Then ' no () else its an exotic word
  438.                     If b$ = "" Then b$ = wd$ Else b$ = b$ + " " + wd$
  439.                 ElseIf InStr(defineWord$(qw$(wd$)), "(") <= 0 Then ' check u insert for q word too
  440.                     If b$ = "" Then b$ = qw$(wd$) Else b$ = b$ + " " + qw$(wd$)
  441.                 End If
  442.             End If
  443.             dp = dp + 1
  444.             If dp > NTopWord Then GoTo fini
  445.             If Timer(.01) - startTime < 0 Then checkTime = Timer(.01) + 24 * 60 * 60 Else checkTime = Timer(.01)
  446.             If checkTime - startTime > timeLimit Then GoTo fini
  447.             Get #1, dp, rec15
  448.             wd$ = _Trim$(rec15)
  449.         Wend
  450.     Next
  451.  
  452.     fini:
  453.     AIwords$ = b$
  454.  
  455. Function qw$ (w$) 'insert the u into a q letter word
  456.     Dim As Long p
  457.     p = InStr(w$, "Q")
  458.     If p Then qw$ = Mid$(w$, 1, p) + "U" + Mid$(w$, p + 1) Else qw$ = w$
  459.  
  460. Function removeRepeats$ (s$) ' s$ is space delimited word list
  461.     ReDim t$(1 To 1), b$
  462.     Dim As Long ub, i, j, ok
  463.     Split s$, " ", t$()
  464.     ub = UBound(t$)
  465.     For i = 1 To ub
  466.         ok = -1
  467.         For j = 1 To i - 1
  468.             If t$(i) = t$(j) Then ok = 0: Exit For
  469.         Next
  470.         If ok Then
  471.             If b$ = "" Then b$ = t$(i) Else b$ = b$ + " " + t$(i)
  472.         End If
  473.     Next
  474.     removeRepeats$ = b$
  475.  
  476. Function score& (a() As String)
  477.     Dim As Long i, s
  478.     For i = 1 To UBound(a)
  479.         Select Case Len(a(i))
  480.             Case 3, 4: s = s + 1
  481.             Case 5: s = s + 2
  482.             Case 6: s = s + 3
  483.             Case 7: s = s + 5
  484.             Case Is > 7: s = s + 11
  485.         End Select
  486.     Next
  487.     score& = s
  488.  
  489. Function defineWord$ (w$) ' this will not edit out definitions that have () in them
  490.     Dim nDef As Long
  491.     w$ = UCase$(w$)
  492.     nDef = Find&(w$)
  493.     If nDef Then
  494.         Open "Collins Words and Defs.RA" For Random As #2 Len = 237
  495.         Get #2, nDef, rec237
  496.         Close #2
  497.     End If
  498.     defineWord$ = _Trim$(rec237)
  499.  

  [ This attachment cannot be displayed inline in 'Print Page' view ]  

Zip with dictionary, wordlist, pointer image and bas source:
Title: Re: Boggle play against AI - WIP
Post by: _vince on January 22, 2022, 12:03:41 am
I think I've got all the bugs and cleaned up:

NICE LOLLIPOP!