/jEdit/tags/jedit-4-3-pre10/org/gjt/sp/jedit/search/SearchAndReplace.java

# · Java · 1391 lines · 886 code · 169 blank · 336 comment · 187 complexity · 61795ced696c27abc4186538c573ea0f MD5 · raw file

  1. /*
  2. * SearchAndReplace.java - Search and replace
  3. * :tabSize=8:indentSize=8:noTabs=false:
  4. * :folding=explicit:collapseFolds=1:
  5. *
  6. * Copyright (C) 1999, 2004 Slava Pestov
  7. * Portions copyright (C) 2001 Tom Locke
  8. *
  9. * This program is free software; you can redistribute it and/or
  10. * modify it under the terms of the GNU General Public License
  11. * as published by the Free Software Foundation; either version 2
  12. * of the License, or any later version.
  13. *
  14. * This program is distributed in the hope that it will be useful,
  15. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  16. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  17. * GNU General Public License for more details.
  18. *
  19. * You should have received a copy of the GNU General Public License
  20. * along with this program; if not, write to the Free Software
  21. * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
  22. */
  23. package org.gjt.sp.jedit.search;
  24. //{{{ Imports
  25. import bsh.*;
  26. import java.awt.*;
  27. import javax.swing.JOptionPane;
  28. import javax.swing.text.Segment;
  29. import org.gjt.sp.jedit.*;
  30. import org.gjt.sp.jedit.gui.TextAreaDialog;
  31. import org.gjt.sp.jedit.io.VFSManager;
  32. import org.gjt.sp.jedit.msg.SearchSettingsChanged;
  33. import org.gjt.sp.jedit.textarea.*;
  34. import org.gjt.sp.util.SegmentCharSequence;
  35. import org.gjt.sp.util.Log;
  36. //}}}
  37. /**
  38. * Class that implements regular expression and literal search within
  39. * jEdit buffers.<p>
  40. *
  41. * There are two main groups of methods in this class:
  42. * <ul>
  43. * <li>Property accessors - for changing search and replace settings.</li>
  44. * <li>Actions - for performing search and replace.</li>
  45. * </ul>
  46. *
  47. * The "HyperSearch" and "Keep dialog" features, as reflected in
  48. * checkbox options in the search dialog, are not handled from within
  49. * this class. If you wish to have these options set before the search dialog
  50. * appears, make a prior call to either or both of the following:
  51. *
  52. * <pre> jEdit.setBooleanProperty("search.hypersearch.toggle",true);
  53. * jEdit.setBooleanProperty("search.keepDialog.toggle",true);</pre>
  54. *
  55. * If you are not using the dialog to undertake a search or replace, you may
  56. * call any of the search and replace methods (including
  57. * {@link #hyperSearch(View)}) without concern for the value of these
  58. * properties.
  59. *
  60. * @author Slava Pestov
  61. * @author John Gellene (API documentation)
  62. * @version $Id: SearchAndReplace.java 9054 2007-03-01 14:48:24Z Vampire0 $
  63. */
  64. public class SearchAndReplace
  65. {
  66. //{{{ Getters and setters
  67. //{{{ setSearchString() method
  68. /**
  69. * Sets the current search string.
  70. * @param search The new search string
  71. */
  72. public static void setSearchString(String search)
  73. {
  74. if(search.equals(SearchAndReplace.search))
  75. return;
  76. SearchAndReplace.search = search;
  77. matcher = null;
  78. EditBus.send(new SearchSettingsChanged(null));
  79. } //}}}
  80. //{{{ getSearchString() method
  81. /**
  82. * Returns the current search string.
  83. */
  84. public static String getSearchString()
  85. {
  86. return search;
  87. } //}}}
  88. //{{{ setReplaceString() method
  89. /**
  90. * Sets the current replacement string.
  91. * @param replace The new replacement string
  92. */
  93. public static void setReplaceString(String replace)
  94. {
  95. if(replace.equals(SearchAndReplace.replace))
  96. return;
  97. SearchAndReplace.replace = replace;
  98. EditBus.send(new SearchSettingsChanged(null));
  99. } //}}}
  100. //{{{ getReplaceString() method
  101. /**
  102. * Returns the current replacement string.
  103. */
  104. public static String getReplaceString()
  105. {
  106. return replace;
  107. } //}}}
  108. //{{{ setIgnoreCase() method
  109. /**
  110. * Sets the ignore case flag.
  111. * @param ignoreCase True if searches should be case insensitive,
  112. * false otherwise
  113. */
  114. public static void setIgnoreCase(boolean ignoreCase)
  115. {
  116. if(ignoreCase == SearchAndReplace.ignoreCase)
  117. return;
  118. SearchAndReplace.ignoreCase = ignoreCase;
  119. matcher = null;
  120. EditBus.send(new SearchSettingsChanged(null));
  121. } //}}}
  122. //{{{ getIgnoreCase() method
  123. /**
  124. * Returns the state of the ignore case flag.
  125. * @return True if searches should be case insensitive,
  126. * false otherwise
  127. */
  128. public static boolean getIgnoreCase()
  129. {
  130. return ignoreCase;
  131. } //}}}
  132. //{{{ setRegexp() method
  133. /**
  134. * Sets the state of the regular expression flag.
  135. * @param regexp True if regular expression searches should be
  136. * performed
  137. */
  138. public static void setRegexp(boolean regexp)
  139. {
  140. if(regexp == SearchAndReplace.regexp)
  141. return;
  142. SearchAndReplace.regexp = regexp;
  143. if(regexp && reverse)
  144. reverse = false;
  145. matcher = null;
  146. EditBus.send(new SearchSettingsChanged(null));
  147. } //}}}
  148. //{{{ getRegexp() method
  149. /**
  150. * Returns the state of the regular expression flag.
  151. * @return True if regular expression searches should be performed
  152. */
  153. public static boolean getRegexp()
  154. {
  155. return regexp;
  156. } //}}}
  157. //{{{ setReverseSearch() method
  158. /**
  159. * Determines whether a reverse search will conducted from the current
  160. * position to the beginning of a buffer. Note that reverse search and
  161. * regular expression search is mutually exclusive; enabling one will
  162. * disable the other.
  163. * @param reverse True if searches should go backwards,
  164. * false otherwise
  165. */
  166. public static void setReverseSearch(boolean reverse)
  167. {
  168. if(reverse == SearchAndReplace.reverse)
  169. return;
  170. SearchAndReplace.reverse = reverse;
  171. EditBus.send(new SearchSettingsChanged(null));
  172. } //}}}
  173. //{{{ getReverseSearch() method
  174. /**
  175. * Returns the state of the reverse search flag.
  176. * @return True if searches should go backwards,
  177. * false otherwise
  178. */
  179. public static boolean getReverseSearch()
  180. {
  181. return reverse;
  182. } //}}}
  183. //{{{ setBeanShellReplace() method
  184. /**
  185. * Sets the state of the BeanShell replace flag.
  186. * @param beanshell True if the replace string is a BeanShell expression
  187. * @since jEdit 3.2pre2
  188. */
  189. public static void setBeanShellReplace(boolean beanshell)
  190. {
  191. if(beanshell == SearchAndReplace.beanshell)
  192. return;
  193. SearchAndReplace.beanshell = beanshell;
  194. EditBus.send(new SearchSettingsChanged(null));
  195. } //}}}
  196. //{{{ getBeanShellReplace() method
  197. /**
  198. * Returns the state of the BeanShell replace flag.
  199. * @return True if the replace string is a BeanShell expression
  200. * @since jEdit 3.2pre2
  201. */
  202. public static boolean getBeanShellReplace()
  203. {
  204. return beanshell;
  205. } //}}}
  206. //{{{ setAutoWrap() method
  207. /**
  208. * Sets the state of the auto wrap around flag.
  209. * @param wrap If true, the 'continue search from start' dialog
  210. * will not be displayed
  211. * @since jEdit 3.2pre2
  212. */
  213. public static void setAutoWrapAround(boolean wrap)
  214. {
  215. if(wrap == SearchAndReplace.wrap)
  216. return;
  217. SearchAndReplace.wrap = wrap;
  218. EditBus.send(new SearchSettingsChanged(null));
  219. } //}}}
  220. //{{{ getAutoWrap() method
  221. /**
  222. * Returns the state of the auto wrap around flag.
  223. * @since jEdit 3.2pre2
  224. */
  225. public static boolean getAutoWrapAround()
  226. {
  227. return wrap;
  228. } //}}}
  229. //{{{ setSearchMatcher() method
  230. /**
  231. * Sets a custom search string matcher. Note that calling
  232. * {@link #setSearchString(String)},
  233. * {@link #setIgnoreCase(boolean)}, or {@link #setRegexp(boolean)}
  234. * will reset the matcher to the default.
  235. */
  236. public static void setSearchMatcher(SearchMatcher matcher)
  237. {
  238. SearchAndReplace.matcher = matcher;
  239. EditBus.send(new SearchSettingsChanged(null));
  240. } //}}}
  241. //{{{ getSearchMatcher() method
  242. /**
  243. * Returns the current search string matcher.
  244. * @return a SearchMatcher or null if there is no search or if the matcher can match empty String
  245. *
  246. * @exception IllegalArgumentException if regular expression search
  247. * is enabled, the search string or replacement string is invalid
  248. * @since jEdit 4.1pre7
  249. */
  250. public static SearchMatcher getSearchMatcher()
  251. throws Exception {
  252. if (matcher != null)
  253. return matcher;
  254. if (search == null || "".equals(search))
  255. return null;
  256. if (regexp)
  257. matcher = new PatternSearchMatcher(search, ignoreCase);
  258. else
  259. matcher = new BoyerMooreSearchMatcher(search, ignoreCase);
  260. if (matcher.nextMatch("", true, true, true, false) != null) {
  261. Log.log(Log.WARNING, SearchAndReplace.class, "The matcher " + matcher + " can match empty string !");
  262. matcher = null;
  263. }
  264. return matcher;
  265. } //}}}
  266. //{{{ setSearchFileSet() method
  267. /**
  268. * Sets the current search file set.
  269. * @param fileset The file set to perform searches in
  270. * @see AllBufferSet
  271. * @see CurrentBufferSet
  272. * @see DirectoryListSet
  273. */
  274. public static void setSearchFileSet(SearchFileSet fileset)
  275. {
  276. SearchAndReplace.fileset = fileset;
  277. EditBus.send(new SearchSettingsChanged(null));
  278. } //}}}
  279. //{{{ getSearchFileSet() method
  280. /**
  281. * Returns the current search file set.
  282. */
  283. public static SearchFileSet getSearchFileSet()
  284. {
  285. return fileset;
  286. } //}}}
  287. //{{{ getSmartCaseReplace() method
  288. /**
  289. * Returns if the replacement string will assume the same case as
  290. * each specific occurrence of the search string.
  291. * @since jEdit 4.2pre10
  292. */
  293. public static boolean getSmartCaseReplace()
  294. {
  295. return (replace != null
  296. && TextUtilities.getStringCase(replace)
  297. == TextUtilities.LOWER_CASE);
  298. } //}}}
  299. //}}}
  300. //{{{ Actions
  301. //{{{ hyperSearch() method
  302. /**
  303. * Performs a HyperSearch.
  304. * @param view The view
  305. * @since jEdit 2.7pre3
  306. */
  307. public static boolean hyperSearch(View view)
  308. {
  309. return hyperSearch(view,false);
  310. } //}}}
  311. //{{{ hyperSearch() method
  312. /**
  313. * Performs a HyperSearch.
  314. * @param view The view
  315. * @param selection If true, will only search in the current selection.
  316. * Note that the file set must be the current buffer file set for this
  317. * to work.
  318. * @since jEdit 4.0pre1
  319. */
  320. public static boolean hyperSearch(View view, boolean selection)
  321. {
  322. // component that will parent any dialog boxes
  323. Component comp = SearchDialog.getSearchDialog(view);
  324. if(comp == null)
  325. comp = view;
  326. record(view,"hyperSearch(view," + selection + ')',false,
  327. !selection);
  328. view.getDockableWindowManager().addDockableWindow(
  329. HyperSearchResults.NAME);
  330. final HyperSearchResults results = (HyperSearchResults)
  331. view.getDockableWindowManager()
  332. .getDockable(HyperSearchResults.NAME);
  333. results.searchStarted();
  334. try
  335. {
  336. SearchMatcher matcher = getSearchMatcher();
  337. if(matcher == null)
  338. {
  339. view.getToolkit().beep();
  340. results.searchFailed();
  341. return false;
  342. }
  343. Selection[] s;
  344. if(selection)
  345. {
  346. s = view.getTextArea().getSelection();
  347. if(s == null)
  348. {
  349. results.searchFailed();
  350. return false;
  351. }
  352. }
  353. else
  354. s = null;
  355. VFSManager.runInWorkThread(new HyperSearchRequest(view,
  356. matcher,results,s));
  357. return true;
  358. }
  359. catch(Exception e)
  360. {
  361. results.searchFailed();
  362. handleError(comp,e);
  363. return false;
  364. }
  365. } //}}}
  366. //{{{ find() method
  367. /**
  368. * Finds the next occurrence of the search string.
  369. * @param view The view
  370. * @return True if the operation was successful, false otherwise
  371. */
  372. public static boolean find(View view)
  373. {
  374. // component that will parent any dialog boxes
  375. Component comp = SearchDialog.getSearchDialog(view);
  376. if(comp == null || !comp.isShowing())
  377. comp = view;
  378. String path = fileset.getNextFile(view,null);
  379. if(path == null)
  380. {
  381. GUIUtilities.error(comp,"empty-fileset",null);
  382. return false;
  383. }
  384. boolean _reverse = reverse && fileset instanceof CurrentBufferSet;
  385. if(_reverse && regexp)
  386. {
  387. GUIUtilities.error(comp,"regexp-reverse",null);
  388. return false;
  389. }
  390. try
  391. {
  392. view.showWaitCursor();
  393. SearchMatcher matcher = getSearchMatcher();
  394. if(matcher == null)
  395. {
  396. view.getToolkit().beep();
  397. return false;
  398. }
  399. record(view,"find(view)",false,true);
  400. boolean repeat = false;
  401. loop: for(;;)
  402. {
  403. while(path != null)
  404. {
  405. Buffer buffer = jEdit.openTemporary(
  406. view,null,path,false);
  407. /* this is stupid and misleading.
  408. * but 'path' is not used anywhere except
  409. * the above line, and if this is done
  410. * after the 'continue', then we will
  411. * either hang, or be forced to duplicate
  412. * it inside the buffer == null, or add
  413. * a 'finally' clause. you decide which one's
  414. * worse. */
  415. path = fileset.getNextFile(view,path);
  416. if(buffer == null)
  417. continue loop;
  418. // Wait for the buffer to load
  419. if(!buffer.isLoaded())
  420. VFSManager.waitForRequests();
  421. int start;
  422. if(view.getBuffer() == buffer && !repeat)
  423. {
  424. JEditTextArea textArea = view.getTextArea();
  425. Selection s = textArea.getSelectionAtOffset(
  426. textArea.getCaretPosition());
  427. if(s == null)
  428. start = textArea.getCaretPosition();
  429. else if(_reverse)
  430. start = s.getStart();
  431. else
  432. start = s.getEnd();
  433. }
  434. else if(_reverse)
  435. start = buffer.getLength();
  436. else
  437. start = 0;
  438. boolean _search = true;
  439. if (!_reverse && matcher.isMatchingEOL())
  440. {
  441. if (start < buffer.getLength())
  442. start += 1;
  443. else
  444. _search = false;
  445. }
  446. if(_search && find(view,buffer,start,repeat,_reverse))
  447. return true;
  448. }
  449. if(repeat)
  450. {
  451. if(!BeanShell.isScriptRunning())
  452. {
  453. view.getStatus().setMessageAndClear(
  454. jEdit.getProperty("view.status.search-not-found"));
  455. view.getToolkit().beep();
  456. }
  457. return false;
  458. }
  459. boolean restart;
  460. // if auto wrap is on, always restart search.
  461. // if auto wrap is off, and we're called from
  462. // a macro, stop search. If we're called
  463. // interactively, ask the user what to do.
  464. if(wrap)
  465. {
  466. if(!BeanShell.isScriptRunning())
  467. {
  468. view.getStatus().setMessageAndClear(
  469. jEdit.getProperty("view.status.auto-wrap"));
  470. // beep if beep property set
  471. if(jEdit.getBooleanProperty("search.beepOnSearchAutoWrap"))
  472. {
  473. view.getToolkit().beep();
  474. }
  475. }
  476. restart = true;
  477. }
  478. else if(BeanShell.isScriptRunning())
  479. {
  480. restart = false;
  481. }
  482. else
  483. {
  484. Integer[] args = {Integer.valueOf(_reverse ? 1 : 0)};
  485. int result = GUIUtilities.confirm(comp,
  486. "keepsearching",args,
  487. JOptionPane.YES_NO_OPTION,
  488. JOptionPane.QUESTION_MESSAGE);
  489. restart = (result == JOptionPane.YES_OPTION);
  490. }
  491. if(restart)
  492. {
  493. // start search from beginning
  494. path = fileset.getFirstFile(view);
  495. repeat = true;
  496. }
  497. else
  498. break loop;
  499. }
  500. }
  501. catch(Exception e)
  502. {
  503. handleError(comp,e);
  504. }
  505. finally
  506. {
  507. view.hideWaitCursor();
  508. }
  509. return false;
  510. } //}}}
  511. //{{{ find() method
  512. /**
  513. * Finds the next instance of the search string in the specified
  514. * buffer.
  515. * @param view The view
  516. * @param buffer The buffer
  517. * @param start Location where to start the search
  518. */
  519. public static boolean find(View view, Buffer buffer, int start)
  520. throws Exception
  521. {
  522. return find(view,buffer,start,false,false);
  523. } //}}}
  524. //{{{ find() method
  525. /**
  526. * Finds the next instance of the search string in the specified
  527. * buffer.
  528. * @param view The view
  529. * @param buffer The buffer
  530. * @param start Location where to start the search
  531. * @param firstTime See {@link SearchMatcher#nextMatch(CharSequence,boolean,boolean,boolean,boolean)}.
  532. * @since jEdit 4.1pre7
  533. */
  534. public static boolean find(View view, Buffer buffer, int start,
  535. boolean firstTime, boolean reverse) throws Exception
  536. {
  537. SearchMatcher matcher = getSearchMatcher();
  538. if(matcher == null)
  539. {
  540. view.getToolkit().beep();
  541. return false;
  542. }
  543. Segment text = new Segment();
  544. if(reverse)
  545. buffer.getText(0,start,text);
  546. else
  547. buffer.getText(start,buffer.getLength() - start,text);
  548. // the start and end flags will be wrong with reverse search enabled,
  549. // but they are only used by the regexp matcher, which doesn't
  550. // support reverse search yet.
  551. //
  552. // REMIND: fix flags when adding reverse regexp search.
  553. SearchMatcher.Match match = matcher.nextMatch(new SegmentCharSequence(text,reverse),
  554. start == 0,true,firstTime,reverse);
  555. if(match != null)
  556. {
  557. jEdit.commitTemporary(buffer);
  558. view.setBuffer(buffer);
  559. JEditTextArea textArea = view.getTextArea();
  560. if(reverse)
  561. {
  562. textArea.setSelection(new Selection.Range(
  563. start - match.end,
  564. start - match.start));
  565. // make sure end of match is visible
  566. textArea.scrollTo(start - match.start,false);
  567. textArea.moveCaretPosition(start - match.end);
  568. }
  569. else
  570. {
  571. textArea.setSelection(new Selection.Range(
  572. start + match.start,
  573. start + match.end));
  574. textArea.moveCaretPosition(start + match.end);
  575. // make sure start of match is visible
  576. textArea.scrollTo(start + match.start,false);
  577. }
  578. return true;
  579. }
  580. else
  581. return false;
  582. } //}}}
  583. //{{{ replace() method
  584. /**
  585. * Replaces the current selection with the replacement string.
  586. * @param view The view
  587. * @return True if the operation was successful, false otherwise
  588. */
  589. public static boolean replace(View view)
  590. {
  591. // component that will parent any dialog boxes
  592. Component comp = SearchDialog.getSearchDialog(view);
  593. if(comp == null)
  594. comp = view;
  595. JEditTextArea textArea = view.getTextArea();
  596. Buffer buffer = view.getBuffer();
  597. if(!buffer.isEditable())
  598. return false;
  599. boolean smartCaseReplace = getSmartCaseReplace();
  600. Selection[] selection = textArea.getSelection();
  601. if (selection.length == 0)
  602. {
  603. try
  604. {
  605. SearchMatcher matcher = getSearchMatcher();
  606. if ((matcher != null) && (matcher.isMatchingEOL()))
  607. {
  608. int caretPosition = textArea.getCaretPosition();
  609. selection = new Selection[] { new Selection.Range(caretPosition,caretPosition) };
  610. }
  611. else
  612. {
  613. view.getToolkit().beep();
  614. return false;
  615. }
  616. }
  617. catch (Exception e)
  618. {
  619. handleError(comp,e);
  620. return false;
  621. }
  622. }
  623. record(view,"replace(view)",true,false);
  624. // a little hack for reverse replace and find
  625. int caret = textArea.getCaretPosition();
  626. Selection s = textArea.getSelectionAtOffset(caret);
  627. if(s != null)
  628. caret = s.getStart();
  629. try
  630. {
  631. buffer.beginCompoundEdit();
  632. SearchMatcher matcher = getSearchMatcher();
  633. if(matcher == null)
  634. return false;
  635. initReplace();
  636. int retVal = 0;
  637. for(int i = 0; i < selection.length; i++)
  638. {
  639. s = selection[i];
  640. retVal += replaceInSelection(view,textArea,
  641. buffer,matcher,smartCaseReplace,s);
  642. }
  643. boolean _reverse = !regexp && reverse && fileset instanceof CurrentBufferSet;
  644. if(_reverse)
  645. {
  646. // so that Replace and Find continues from
  647. // the right location
  648. textArea.moveCaretPosition(caret);
  649. }
  650. else
  651. {
  652. s = textArea.getSelectionAtOffset(
  653. textArea.getCaretPosition());
  654. if(s != null)
  655. textArea.moveCaretPosition(s.getEnd());
  656. }
  657. if(!BeanShell.isScriptRunning())
  658. {
  659. Object[] args = {Integer.valueOf(retVal),
  660. Integer.valueOf(1)};
  661. view.getStatus().setMessageAndClear(jEdit.getProperty(
  662. "view.status.replace-all",args));
  663. }
  664. if(retVal == 0)
  665. {
  666. view.getToolkit().beep();
  667. return false;
  668. }
  669. return true;
  670. }
  671. catch(Exception e)
  672. {
  673. handleError(comp,e);
  674. }
  675. finally
  676. {
  677. buffer.endCompoundEdit();
  678. }
  679. return false;
  680. } //}}}
  681. //{{{ replace() method
  682. /**
  683. * Replaces text in the specified range with the replacement string.
  684. * @param view The view
  685. * @param buffer The buffer
  686. * @param start The start offset
  687. * @param end The end offset
  688. * @return True if the operation was successful, false otherwise
  689. */
  690. public static boolean replace(View view, Buffer buffer, int start, int end)
  691. {
  692. if(!buffer.isEditable())
  693. return false;
  694. // component that will parent any dialog boxes
  695. Component comp = SearchDialog.getSearchDialog(view);
  696. if(comp == null)
  697. comp = view;
  698. boolean smartCaseReplace = getSmartCaseReplace();
  699. try
  700. {
  701. buffer.beginCompoundEdit();
  702. SearchMatcher matcher = getSearchMatcher();
  703. if(matcher == null)
  704. return false;
  705. initReplace();
  706. int retVal = 0;
  707. retVal += _replace(view,buffer,matcher,start,end,
  708. smartCaseReplace);
  709. if(retVal != 0)
  710. return true;
  711. }
  712. catch(Exception e)
  713. {
  714. handleError(comp,e);
  715. }
  716. finally
  717. {
  718. buffer.endCompoundEdit();
  719. }
  720. return false;
  721. } //}}}
  722. //{{{ replaceAll() method
  723. /**
  724. * Replaces all occurrences of the search string with the replacement
  725. * string.
  726. * @param view The view
  727. */
  728. public static boolean replaceAll(View view)
  729. {
  730. return replaceAll(view,false);
  731. } //}}}
  732. //{{{ replaceAll() method
  733. /**
  734. * Replaces all occurrences of the search string with the replacement
  735. * string.
  736. * @param view The view
  737. * @param dontOpenChangedFiles Whether to open changed files or to autosave them quietly
  738. */
  739. public static boolean replaceAll(View view, boolean dontOpenChangedFiles)
  740. {
  741. // component that will parent any dialog boxes
  742. Component comp = SearchDialog.getSearchDialog(view);
  743. if(comp == null)
  744. comp = view;
  745. if(fileset.getFileCount(view) == 0)
  746. {
  747. GUIUtilities.error(comp,"empty-fileset",null);
  748. return false;
  749. }
  750. record(view,"replaceAll(view)",true,true);
  751. view.showWaitCursor();
  752. boolean smartCaseReplace = (replace != null
  753. && TextUtilities.getStringCase(replace)
  754. == TextUtilities.LOWER_CASE);
  755. int fileCount = 0;
  756. int occurCount = 0;
  757. try
  758. {
  759. SearchMatcher matcher = getSearchMatcher();
  760. if(matcher == null)
  761. return false;
  762. initReplace();
  763. String path = fileset.getFirstFile(view);
  764. loop: while(path != null)
  765. {
  766. Buffer buffer = jEdit.openTemporary(
  767. view,null,path,false);
  768. /* this is stupid and misleading.
  769. * but 'path' is not used anywhere except
  770. * the above line, and if this is done
  771. * after the 'continue', then we will
  772. * either hang, or be forced to duplicate
  773. * it inside the buffer == null, or add
  774. * a 'finally' clause. you decide which one's
  775. * worse. */
  776. path = fileset.getNextFile(view,path);
  777. if(buffer == null)
  778. continue loop;
  779. // Wait for buffer to finish loading
  780. if(buffer.isPerformingIO())
  781. VFSManager.waitForRequests();
  782. if(!buffer.isEditable())
  783. continue loop;
  784. // Leave buffer in a consistent state if
  785. // an error occurs
  786. int retVal = 0;
  787. try
  788. {
  789. buffer.beginCompoundEdit();
  790. retVal = _replace(view,buffer,matcher,
  791. 0,buffer.getLength(),
  792. smartCaseReplace);
  793. }
  794. finally
  795. {
  796. buffer.endCompoundEdit();
  797. }
  798. if(retVal != 0)
  799. {
  800. fileCount++;
  801. occurCount += retVal;
  802. if (dontOpenChangedFiles)
  803. {
  804. buffer.save(null,null);
  805. }
  806. else
  807. {
  808. jEdit.commitTemporary(buffer);
  809. }
  810. }
  811. }
  812. }
  813. catch(Exception e)
  814. {
  815. handleError(comp,e);
  816. }
  817. finally
  818. {
  819. view.hideWaitCursor();
  820. }
  821. /* Don't do this when playing a macro, cos it's annoying */
  822. if(!BeanShell.isScriptRunning())
  823. {
  824. Object[] args = {Integer.valueOf(occurCount),
  825. Integer.valueOf(fileCount)};
  826. view.getStatus().setMessageAndClear(jEdit.getProperty(
  827. "view.status.replace-all",args));
  828. if(occurCount == 0)
  829. view.getToolkit().beep();
  830. }
  831. return (fileCount != 0);
  832. } //}}}
  833. //}}}
  834. //{{{ escapeRegexp() method
  835. /**
  836. * Escapes characters with special meaning in a regexp.
  837. * @param multiline Should \n be escaped?
  838. * @since jEdit 4.3pre1
  839. */
  840. public static String escapeRegexp(String str, boolean multiline)
  841. {
  842. return MiscUtilities.charsToEscapes(str,
  843. "\r\t\\()[]{}$^*+?|."
  844. + (multiline ? "" : "\n"));
  845. } //}}}
  846. //{{{ load() method
  847. /**
  848. * Loads search and replace state from the properties.
  849. */
  850. public static void load()
  851. {
  852. search = jEdit.getProperty("search.find.value");
  853. replace = jEdit.getProperty("search.replace.value");
  854. ignoreCase = jEdit.getBooleanProperty("search.ignoreCase.toggle");
  855. regexp = jEdit.getBooleanProperty("search.regexp.toggle");
  856. beanshell = jEdit.getBooleanProperty("search.beanshell.toggle");
  857. wrap = jEdit.getBooleanProperty("search.wrap.toggle");
  858. fileset = new CurrentBufferSet();
  859. // Tags plugin likes to call this method at times other than
  860. // startup; so we need to fire a SearchSettingsChanged to
  861. // notify the search bar and so on.
  862. matcher = null;
  863. EditBus.send(new SearchSettingsChanged(null));
  864. } //}}}
  865. //{{{ save() method
  866. /**
  867. * Saves search and replace state to the properties.
  868. */
  869. public static void save()
  870. {
  871. jEdit.setProperty("search.find.value",search);
  872. jEdit.setProperty("search.replace.value",replace);
  873. jEdit.setBooleanProperty("search.ignoreCase.toggle",ignoreCase);
  874. jEdit.setBooleanProperty("search.regexp.toggle",regexp);
  875. jEdit.setBooleanProperty("search.beanshell.toggle",beanshell);
  876. jEdit.setBooleanProperty("search.wrap.toggle",wrap);
  877. } //}}}
  878. //{{{ handleError() method
  879. static void handleError(Component comp, Exception e)
  880. {
  881. Log.log(Log.ERROR,SearchAndReplace.class,e);
  882. if(comp instanceof Dialog)
  883. {
  884. new TextAreaDialog((Dialog)comp,
  885. beanshell ? "searcherror-bsh"
  886. : "searcherror",e);
  887. }
  888. else
  889. {
  890. new TextAreaDialog((Frame)comp,
  891. beanshell ? "searcherror-bsh"
  892. : "searcherror",e);
  893. }
  894. } //}}}
  895. //{{{ Private members
  896. //{{{ Instance variables
  897. private static String search;
  898. private static String replace;
  899. private static BshMethod replaceMethod;
  900. private static NameSpace replaceNS = new NameSpace(
  901. BeanShell.getNameSpace(),
  902. BeanShell.getNameSpace().getClassManager(),
  903. "search and replace");
  904. private static boolean regexp;
  905. private static boolean ignoreCase;
  906. private static boolean reverse;
  907. private static boolean beanshell;
  908. private static boolean wrap;
  909. private static SearchMatcher matcher;
  910. private static SearchFileSet fileset;
  911. //}}}
  912. //{{{ initReplace() method
  913. /**
  914. * Set up BeanShell replace if necessary.
  915. */
  916. private static void initReplace() throws Exception
  917. {
  918. if(beanshell && replace.length() != 0)
  919. {
  920. replaceMethod = BeanShell.cacheBlock("replace",
  921. "return (" + replace + ");",true);
  922. }
  923. else
  924. replaceMethod = null;
  925. } //}}}
  926. //{{{ record() method
  927. private static void record(View view, String action,
  928. boolean replaceAction, boolean recordFileSet)
  929. {
  930. Macros.Recorder recorder = view.getMacroRecorder();
  931. if(recorder != null)
  932. {
  933. recorder.record("SearchAndReplace.setSearchString(\""
  934. + MiscUtilities.charsToEscapes(search) + "\");");
  935. if(replaceAction)
  936. {
  937. recorder.record("SearchAndReplace.setReplaceString(\""
  938. + MiscUtilities.charsToEscapes(replace) + "\");");
  939. recorder.record("SearchAndReplace.setBeanShellReplace("
  940. + beanshell + ");");
  941. }
  942. else
  943. {
  944. // only record this if doing a find next
  945. recorder.record("SearchAndReplace.setAutoWrapAround("
  946. + wrap + ");");
  947. recorder.record("SearchAndReplace.setReverseSearch("
  948. + reverse + ");");
  949. }
  950. recorder.record("SearchAndReplace.setIgnoreCase("
  951. + ignoreCase + ");");
  952. recorder.record("SearchAndReplace.setRegexp("
  953. + regexp + ");");
  954. if(recordFileSet)
  955. {
  956. recorder.record("SearchAndReplace.setSearchFileSet("
  957. + fileset.getCode() + ");");
  958. }
  959. recorder.record("SearchAndReplace." + action + ';');
  960. }
  961. } //}}}
  962. //{{{ replaceInSelection() method
  963. private static int replaceInSelection(View view, JEditTextArea textArea,
  964. Buffer buffer, SearchMatcher matcher, boolean smartCaseReplace,
  965. Selection s) throws Exception
  966. {
  967. /* if an occurence occurs at the
  968. beginning of the selection, the
  969. selection start will get moved.
  970. this sucks, so we hack to avoid it. */
  971. int start = s.getStart();
  972. int returnValue;
  973. if(s instanceof Selection.Range)
  974. {
  975. returnValue = _replace(view,buffer,matcher,
  976. s.getStart(),s.getEnd(),
  977. smartCaseReplace);
  978. textArea.removeFromSelection(s);
  979. textArea.addToSelection(new Selection.Range(
  980. start,s.getEnd()));
  981. }
  982. else if(s instanceof Selection.Rect)
  983. {
  984. Selection.Rect rect = (Selection.Rect)s;
  985. int startCol = rect.getStartColumn(
  986. buffer);
  987. int endCol = rect.getEndColumn(
  988. buffer);
  989. returnValue = 0;
  990. for(int j = s.getStartLine(); j <= s.getEndLine(); j++)
  991. {
  992. returnValue += _replace(view,buffer,matcher,
  993. getColumnOnOtherLine(buffer,j,startCol),
  994. getColumnOnOtherLine(buffer,j,endCol),
  995. smartCaseReplace);
  996. }
  997. textArea.addToSelection(new Selection.Rect(
  998. start,s.getEnd()));
  999. }
  1000. else
  1001. throw new RuntimeException("Unsupported: " + s);
  1002. return returnValue;
  1003. } //}}}
  1004. //{{{ _replace() method
  1005. /**
  1006. * Replaces all occurrences of the search string with the replacement
  1007. * string.
  1008. * @param view The view
  1009. * @param buffer The buffer
  1010. * @param start The start offset
  1011. * @param end The end offset
  1012. * @param matcher The search matcher to use
  1013. * @param smartCaseReplace See user's guide
  1014. * @return The number of occurrences replaced
  1015. */
  1016. private static int _replace(View view, Buffer buffer,
  1017. SearchMatcher matcher, int start, int end,
  1018. boolean smartCaseReplace)
  1019. throws Exception
  1020. {
  1021. int occurCount = 0;
  1022. boolean endOfLine = (buffer.getLineEndOffset(
  1023. buffer.getLineOfOffset(end)) - 1 == end);
  1024. Segment text = new Segment();
  1025. int offset = start;
  1026. loop: for(int counter = 0; ; counter++)
  1027. {
  1028. buffer.getText(offset,end - offset,text);
  1029. boolean startOfLine = (buffer.getLineStartOffset(
  1030. buffer.getLineOfOffset(offset)) == offset);
  1031. SearchMatcher.Match occur = matcher.nextMatch(
  1032. new SegmentCharSequence(text,false),
  1033. startOfLine,endOfLine,counter == 0,
  1034. false);
  1035. if(occur == null)
  1036. break loop;
  1037. String found = new String(text.array,
  1038. text.offset + occur.start,
  1039. occur.end - occur.start);
  1040. int length = replaceOne(view,buffer,occur,offset,
  1041. found,smartCaseReplace);
  1042. if(length == -1)
  1043. offset += occur.end;
  1044. else
  1045. {
  1046. offset += occur.start + length;
  1047. end += (length - found.length());
  1048. occurCount++;
  1049. }
  1050. if (matcher.isMatchingEOL())
  1051. {
  1052. if (offset < buffer.getLength())
  1053. offset += 1;
  1054. else
  1055. break loop;
  1056. if (offset >= end)
  1057. break loop;
  1058. }
  1059. }
  1060. return occurCount;
  1061. } //}}}
  1062. //{{{ replaceOne() method
  1063. /**
  1064. * Replace one occurrence of the search string with the
  1065. * replacement string.
  1066. */
  1067. private static int replaceOne(View view, Buffer buffer,
  1068. SearchMatcher.Match occur, int offset, String found,
  1069. boolean smartCaseReplace)
  1070. throws Exception
  1071. {
  1072. String subst = replaceOne(view,occur,found);
  1073. if(smartCaseReplace && ignoreCase)
  1074. {
  1075. int strCase = TextUtilities.getStringCase(found);
  1076. if(strCase == TextUtilities.LOWER_CASE)
  1077. subst = subst.toLowerCase();
  1078. else if(strCase == TextUtilities.UPPER_CASE)
  1079. subst = subst.toUpperCase();
  1080. else if(strCase == TextUtilities.TITLE_CASE)
  1081. subst = TextUtilities.toTitleCase(subst);
  1082. }
  1083. if(subst != null)
  1084. {
  1085. int start = offset + occur.start;
  1086. int end = offset + occur.end;
  1087. if (end - start > 0)
  1088. buffer.remove(start,end - start);
  1089. buffer.insert(start,subst);
  1090. return subst.length();
  1091. }
  1092. else
  1093. return -1;
  1094. } //}}}
  1095. //{{{ replaceOne() method
  1096. private static String replaceOne(View view,
  1097. SearchMatcher.Match occur, String found)
  1098. throws Exception
  1099. {
  1100. if(regexp)
  1101. {
  1102. if(replaceMethod != null)
  1103. return regexpBeanShellReplace(view,occur);
  1104. else
  1105. return regexpReplace(occur,found);
  1106. }
  1107. else
  1108. {
  1109. if(replaceMethod != null)
  1110. return literalBeanShellReplace(view,found);
  1111. else
  1112. return replace;
  1113. }
  1114. } //}}}
  1115. //{{{ regexpBeanShellReplace() method
  1116. private static String regexpBeanShellReplace(View view,
  1117. SearchMatcher.Match occur) throws Exception
  1118. {
  1119. for(int i = 0; i < occur.substitutions.length; i++)
  1120. {
  1121. replaceNS.setVariable("_" + i,
  1122. occur.substitutions[i]);
  1123. }
  1124. Object obj = BeanShell.runCachedBlock(
  1125. replaceMethod,view,replaceNS);
  1126. if(obj == null)
  1127. return "";
  1128. else
  1129. return obj.toString();
  1130. } //}}}
  1131. //{{{ regexpReplace() method
  1132. private static String regexpReplace(SearchMatcher.Match occur,
  1133. String found) throws Exception
  1134. {
  1135. StringBuilder buf = new StringBuilder();
  1136. for(int i = 0; i < replace.length(); i++)
  1137. {
  1138. char ch = replace.charAt(i);
  1139. switch(ch)
  1140. {
  1141. case '$':
  1142. if(i == replace.length() - 1)
  1143. {
  1144. buf.append(ch);
  1145. break;
  1146. }
  1147. ch = replace.charAt(++i);
  1148. if(ch == '$')
  1149. buf.append('$');
  1150. else if(ch == '0')
  1151. buf.append(found);
  1152. else if(Character.isDigit(ch))
  1153. {
  1154. int n = ch - '0';
  1155. if(n < occur
  1156. .substitutions
  1157. .length)
  1158. {
  1159. buf.append(
  1160. occur
  1161. .substitutions
  1162. [n]
  1163. );
  1164. }
  1165. }
  1166. break;
  1167. case '\\':
  1168. if(i == replace.length() - 1)
  1169. {
  1170. buf.append('\\');
  1171. break;
  1172. }
  1173. ch = replace.charAt(++i);
  1174. switch(ch)
  1175. {
  1176. case 'n':
  1177. buf.append('\n');
  1178. break;
  1179. case 't':
  1180. buf.append('\t');
  1181. break;
  1182. default:
  1183. buf.append(ch);
  1184. break;
  1185. }
  1186. break;
  1187. default:
  1188. buf.append(ch);
  1189. break;
  1190. }
  1191. }
  1192. return buf.toString();
  1193. } //}}}
  1194. //{{{ literalBeanShellReplace() method
  1195. private static String literalBeanShellReplace(View view, String found)
  1196. throws Exception
  1197. {
  1198. replaceNS.setVariable("_0",found);
  1199. Object obj = BeanShell.runCachedBlock(
  1200. replaceMethod,
  1201. view,replaceNS);
  1202. if(obj == null)
  1203. return "";
  1204. else
  1205. return obj.toString();
  1206. } //}}}
  1207. //{{{ getColumnOnOtherLine() method
  1208. /**
  1209. * Should be somewhere else...
  1210. */
  1211. private static int getColumnOnOtherLine(Buffer buffer, int line,
  1212. int col)
  1213. {
  1214. int returnValue = buffer.getOffsetOfVirtualColumn(
  1215. line,col,null);
  1216. if(returnValue == -1)
  1217. return buffer.getLineEndOffset(line) - 1;
  1218. else
  1219. return buffer.getLineStartOffset(line) + returnValue;
  1220. } //}}}
  1221. //}}}
  1222. }