The problem is that the state doesn’t change immediately. I believe it requires at least a dispatch() of some sort, but adding dispatch(state.tr); right before replaceNext() will not help either – the state stays unchanged.
So I’m at loss how to get the state changed before replaceNext() and do the replace with the correct replace term.
Someone (Claude & Grok) also suggested this, but this won’t replace text at all:
replace: (replaceTerm) => ({ state, dispatch, view }) => {
const searchState = getSearchState(state);
if (!searchState) return false;
// Create a new SearchQuery with the replace term
const query = new SearchQuery({
search: searchState.query.search,
replace: replaceTerm,
caseSensitive: searchState.query.caseSensitive,
regexp: searchState.query.regexp,
wholeWord: searchState.query.wholeWord,
});
if (dispatch) {
// Create a transaction and set the new search state
const tr = state.tr;
setSearchState(tr, query);
// Dispatch the transaction to update the state
dispatch(tr);
// Apply the transaction to get the updated state
const newState = state.apply(tr);
// Now call replaceNext with the updated state
return replaceNext(newState, dispatch);
}
// If no dispatch, just return the result of replaceNext with original state
return replaceNext(state);
},
This also won’t have the correct replace term in replaceNext(). In replaceNext(), the query that it gets is the previous one (the one that exists before setSearchState is called):
It seems that Tiptap likes to call dispatch on its own, and I should not intervene. So I solved this by splitting the setSearchState and replaceNext in separate commands: