I'm new to Python and while I have some experience with Java, I haven't coded in a long time. I'm currently trying to adapt a word search solver from wordsearch_solver by aphrx to find offensive words in a puzzle and append them to a list. I need a little help with the logic.
The solver should be part of a word search puzzle generator I already have. The solver works by checking "if word in row" (horizontally, vertically or diagonally) and appending the word to a list (fWords) if it's true. While/as long as the list (fWords) contains words, it should regenerate the word search puzzle.
However, I've encountered the problem that some offensive words are often part of words that were purposely placed in the puzzle, e.g. hell in hello, tit in title or ass in assemble. Since these words were purposely placed in the puzzle they will always be part of it and the code gets stuck in the loop regenerating the puzzle till the list of offensive words is empty which it never will be.
I've provided a sample grid (grid), a list of words hidden in the puzzle (wordlist) and a list of offensive words (badwords) that should be avoided. In my code the loop it gets stuck in is under "make_puzzle".For better demonstration I also have a screenshot of the grid/puzzle:
https://imgur.com/a/NJ6cxjb
1. Check if a word from badwords is in the current row being checked.
2. Check if a word_wordlist from wordlist is in the current row being checked.
I've tried the following code in different variations:
Other approaches I've thought of are appending the row to a temporary list if the row contains a word_wordlist from wordlist and then skipping rows that are identical to the items in the temporary list. A problem here then though is that the row could still contain other offensive words and by skipping that row, they would not be flagged/added to the list fWords, e.g. a b c T I T e f g H E L L O. Since this row contains hello, it would get skipped and tit would not get added to fWords.
I've been working on this for around a week trying different variations and approaches but I think I might be approaching this wrong. Any guidance is appreciated.
The solver should be part of a word search puzzle generator I already have. The solver works by checking "if word in row" (horizontally, vertically or diagonally) and appending the word to a list (fWords) if it's true. While/as long as the list (fWords) contains words, it should regenerate the word search puzzle.
However, I've encountered the problem that some offensive words are often part of words that were purposely placed in the puzzle, e.g. hell in hello, tit in title or ass in assemble. Since these words were purposely placed in the puzzle they will always be part of it and the code gets stuck in the loop regenerating the puzzle till the list of offensive words is empty which it never will be.
I've provided a sample grid (grid), a list of words hidden in the puzzle (wordlist) and a list of offensive words (badwords) that should be avoided. In my code the loop it gets stuck in is under "make_puzzle".For better demonstration I also have a screenshot of the grid/puzzle:
grid = [[ 'IZCTFXORREKETTRHSKZD'], ['HZADCBYLXYMWOENGYHXI'], ['MUHLSFGHHTIULSPUYKPN'], ['IYKXCJFKTWOEENOCQGGD'], ['NNWBGTOKLSHZGZSZFFVW'], ['MPIHEAHJMLJHUPLADQWB'], ['OWIRTPLMZJYBGZGXQHDD'], ['APOLUARPVYBRLWTFUDMJ'], ['OLPZROPYQUQHKBNKEUKK'], ['WRMZQPFGVDNDNRBDHOHW'], ['WAUSQEZEWTZVZQIEIQVQ'], ['QLCTRHLCBLAYCJLZZLMN'], ['OCCBUDTLJFDXJLMKVNEF'], ['DBQHGLEMPTHXOAQVMZDM'], ['RYGFSRIRTGIICICTGIPZ'], ['DAECZQHDDCETDRSTOEZH'], ['NVUNGMSHTWTFLQXBTCQK'], ['UIDEOZMOEYRBSEPFBYOC'], ['ZNDCJEHVUAKXTSIMLVOG'], ['ZDIPBYTCWWQTHUCMSEPN']] badwords = ['TIT', 'HELL', 'WTF'] wordlist = ['HELLO', 'TITLE', 'FOUR', 'ONE', 'TWO'] fWords = [] # nrows is hard-coded in this example because the grid doesn't change nrows = 20 def check_word(row, word, wlist, dir): if word in str(row): print(word + ' is in ' + str(row) + ' Direction: ' + dir) fWords.append(word) return True def find_word(p2, bwords, rows, wlist): for word in bwords: find_horizontal(p2, word, wlist) find_vertical(p2, word, wlist) find_diagonal(p2, word, rows, wlist) def find_horizontal(p2, word, wlist): dir = 'horizontal' dir_r = 'horizontal_reverse' for row in enumerate(p2): check_word(str(row), word, wlist, dir) row_r = str(row)[::-1] check_word(str(row_r), word, wlist, dir_r) return False def find_vertical(p2, word, wlist): dir = 'vertical' dir_r = 'vertical_reverse' for char in range(len(p2[0][0])): temp = [] for col in range(len(p2)): temp.append(p2[col][0][char]) temp = ''.join(temp) temp_r = temp[::-1] check_word(str(temp), word, wlist, dir) check_word(str(temp_r), word, wlist, dir_r) return False def find_diagonal(p2, word, rows, wlist): dir = 'diagonal' for a in range(0, len(p2[0][0])): temp = [[] for i in range(8)] ranges = [[] for i in range(8)] i = 0 while ((a - i) >= 0) and (i < len(p2)): coords = [[i, a - i], [(rows - 1) - i, a - i], [(rows - 1) - i, (rows - 1) - (a - i)], [i, (rows - 1) - (a - i)]] for cx, c in enumerate(coords): temp[cx].append(p2[c[0]][0][c[1]]) ranges[cx].append((c[0], c[1])) ranges[cx + 4].append((c[1], c[0])) i += 1 for ti in range(4): temp[ti] = ''.join(temp[ti]) temp[ti + 4] = temp[ti][::-1] for t in enumerate(temp): check_word(str(t), word, wlist, dir) return False def checker(grid, nrows, wlist): find_word(grid, badwords, nrows, wlist) return fWords # START OF THE PROGRAM # def _make_puzzle(...) checker(grid, nrows, wordlist) # def make_puzzle(...) - This is where the puzzle loops while fWords: print(fWords) fWords.clear() # The puzzle grid would get regenerated here: grid = _make_puzzle(*args, **kwargs) if not fWords: if grid: pass # The final grid without offensive words would be returned here: return gridOutput:
Output:TIT is in (1, 'MIELTITJCZPRLIPN') Direction: diagonal
TIT is in (5, 'NPILRPZCJTITLEIM') Direction: diagonal
HELL is in (6, 'DMUHELLOIEWEVT') Direction: diagonal
WTF is in (7, ['APOLUARPVYBRLWTFUDMJ']) Direction: horizontal
WTF is in (16, ['NVUNGMSHTWTFLQXBTCQK']) Direction: horizontal
['TIT', 'TIT', 'HELL', 'WTF', 'WTF']
The part that needs fixing is def check_word. The logic of how to implement the changes I want is a little beyond me. I got a little lost with all the nested loops and if statements. I think I have to1. Check if a word from badwords is in the current row being checked.
2. Check if a word_wordlist from wordlist is in the current row being checked.
I've tried the following code in different variations:
def check_word(row, word, wlist, dir): if word in str(row): for word_wordlist in wlist: if (word not in word_wordlist) or (word not in word_wordlist[::-1]): print(word + ' is in ' + str(row) + ' Direction: ' + dir) fWords.append(word) return True elif (word in word_wordlist) or (word in word_wordlist[::-1]): continueAlso:
def check_word(row, word, wlist, dir): for word_wordlist in wlist: if word_wordlist in str(row): continue if (word_wordlist not in str(row)) and (word in str(row)): print(word + ' is in ' + str(row) + ' Direction: ' + dir) fWords.append(word) return TrueI've tried rearranging the loops and the if statements but the nesting is a little confusing to me. Essentially, if a word is found as part of a word_wordlist, it should skip that word and continue with the next word but I'm not sure how to implement that properly.
Other approaches I've thought of are appending the row to a temporary list if the row contains a word_wordlist from wordlist and then skipping rows that are identical to the items in the temporary list. A problem here then though is that the row could still contain other offensive words and by skipping that row, they would not be flagged/added to the list fWords, e.g. a b c T I T e f g H E L L O. Since this row contains hello, it would get skipped and tit would not get added to fWords.
I've been working on this for around a week trying different variations and approaches but I think I might be approaching this wrong. Any guidance is appreciated.