pythonscript: any ready pyscript to replace one huge set of regex/ phrases with others?
- 
 @Alan-Kilborn 
 just tested and it works like magic. 👍little question about the python script plugin: 
 do you, by any chance, know of any hidden button or hotkey, to directly open the path to the user scripts at …\plugins\Config\PythonScript\scripts from within the plugin’s menu items, instead of browsing there manually ?
- 
 @Meta-Chuh said: do you, by any chance, know of any hidden button or hotkey, to directly open the path to the user scripts at …, instead of browsing there manually ? Hmmm, not really…I guess if I wanted that I would make it a Favorite in the Explorer plugin. :) 
- 
 As I learned a few months ago (it’s buried in the docs, but I needed other users to point it out to me): to edit an existing script, instead of clicking normally on the script, use Ctrl+Click. After that, you can use the tabbar > right-click entry to open that directory in Explorer, if you find that easier for opening other scripts from the same folder. update: https://github.com/bruderstein/PythonScript/issues/108 was where I learned this. 
- 
 Since you have a solution that works, why go looking for another one? 
 My data is in npp. I have to copy-paste it to firefox and run foxscript, and bring the processed data back to npp.Thanks. 
- 
 wow. so simple. thanks. in my case, ctrl-leftclick on the script name in the menu, opens that script in npp to edit. rightclick doesn’t do anything. 
- 
 Yes, but seems like @Meta-Chuh was looking for something more direct. 
- 
 @PeterJones said: use Ctrl+Click thats even better, thanks. 
 editing/viewing existing scripts is the thing i’ll probably use most in the future i suppose.btw: i just read your old thread and i just had to copy the following sentence and apply it unchanged to myself ;-) Well, I (me, metachuh) feel silly. It’s documented right there in the Installation: Usage in the help file: “To edit the script, just hold Ctrl down and click the script in the Scripts menu”. I guess that proves I’ve only read the help file when looking for specific commands. Someday, I should really read the whole thing. :-) 
- 
 Hi, @alan-kilborn, I cannot get your Python script ( see below ) running correctly ! I certainly misunderstood what it is for :-(( To my mind, assuming your list, it should replace any lower-case letter a,bandcwith the corresponding upper-case letter, in current file, shouldn’t it ?# format for each line is: delimiter then search regex then delimiter then replace regex sr_list = [ '!a!A', '@b@B', '!c!C', ] # or take input from a file: #with open(r'sr_list.txt') as f: sr_list = f.readlines() editor.beginUndoAction() for line in sr_list: (s,r) = line[1:].rstrip().split(line[0]) editor.rereplace(s,r) editor.endUndoAction()I tested your script, with Python script v1.2.0.0and Notepad++v6.0and here is my debug info :Notepad++ v7.6 (32-bit) Build time : Nov 12 2018 - 23:51:42 Path : D:\@@\760\notepad++.exe Admin mode : OFF Local Conf mode : ON OS : Windows XP (32-bit) Plugins : BetterMultiSelection.dll ComparePlugin.dll CustomizeToolbar.dll DSpellCheck.dll LightExplorer.dll LuaScript.dll mimeTools.dll NppConverter.dll NppExec.dll NppExport.dll NppTextFX.dll PythonScript.dllOn the other hand, my Python installation seems correct as, for instance, your other script, below, works nice, highlighting the strings Hello there, in light blue Mark style 1
 So, Alan, what obvious thing am I missing ?! Best Regards guy038 
- 
 I am sorry to hear it didn’t work for you. :-( It is really a simple script, I’m not sure what could go wrong. Maybe try putting this line before the rereplace line and perhaps it will provide a hint of some sort: console.write('s:{} r:{}\n'.format(s,r))Also, but this would have nothing to do with it, the strings in sr_list should be raw strings for best use with regex without getting “LTS”. Thus, '!a!A'should be'r!a!A'and same for any other strings put into the sr_list.
- 
 what happens if you try a complete new pythonscript 1.3.0.0 install on a new 7.6.2 portable ? 
 still the same, or does alan’s script work on that ?i’ve added the folder locations, from the other py thread, below as convenience if lib location or anything else of the add ons might be a trigger: get a new copy of PythonScript_Full_1.3.0.0.zip from here 
 extract it and put it’s contents as listed belowPythonScript.dll, plugin dll goes to: 
 npp.7.6.2.bin\plugins\PythonScript\PythonScript.dllpython27.dll goes to: 
 npp.7.6.2.bin\python27.dllmachine level scripts and python library go to: 
 npp.7.6.2.bin\plugins\PythonScript\lib\
 npp.7.6.2.bin\plugins\PythonScript\scripts\
 contains sample scripts and startup. pymanual, context-help files go to: 
 npp.7.6.2.bin\plugins\doc\PythonScript\
 contains PythonScript.chm up to version 1.2.0.0
 contains html docs since version 1.3.0.0user level scripts go to: 
 npp.7.6.2.bin\plugins\config\PythonScript\scripts\
 note: this folder will be created automatically as soon as a new script is created.
- 
 @Alan-Kilborn said: It is really a simple script it may be simple, but it definitively comes in very handy. 
 i use it, love it, and without you i wouldn’t have it. 😃👍before using your script, i used your suggested way of using macros for multi pass, multi regexes on files. 
 (and before that, there was pure darkness 😂 )but now i prefer your script, as the regexes are much easier to change or read than within a saved macro. 👍👍👍 
- 
 Hello @alan-kilborn, @meta-chuh and All, Sorry, I preferred to take some time, doing numerous tests and … it works nicely ;-)) Alan, it’s just my mistake, because I should have opened the console, immediately ! Indeed, I wrote some accentuated characters above \x7f, although in a comment ! So I added the directive#coding=utf-8as first line of the script. You’re really lucky as an[A-Z]person ;-))In order to run S/R in an insensitive way, I just imported the relibrary and used the flagre.IGNORECASEI also tried your second solution with strings/regexes in a file and… no problem, too ! Just notice that, in this case, the exact python code is, rather : f = open(r'<Drive_Letter>:\....\....\sr_list.txt') sr_list = f.readlines()I, first thought that, according your comments, the part, to be inserted, was, literally : open(r'sr_list.txt') as f: sr_list = f.readlines()But, of course, this code is wrong and an error occurs on word as-:))
 Hi, @meta-chuh, Hummmm…, I’m hungry, since a while… So be patient a bit ! I be back very soon and, as I’ve already installed a portable v7.6.2version of N++, I’ll, simply, need to add the latest PythonJust before posting, I’ve seen your last reply to Alan and I do agree to your compliments ! It’s really a magic script ;-)) Best Regards, guy038 
- 
 An aside on as:@guy038 said: I, first thought that, according your comments, the part, to be inserted, was, literally : 
 open(r'sr_list.txt') as f: sr_list = f.readlines()@Alan-Kilborn said: with open(r’sr_list.txt’) as f: sr_list = f.readlines() Having just done some Python tutorials, I learned about the withstatement, which is what enables theas– thewithwas literally part of what was needed in order for theasto work
- 
 @guy038 said: It’s really a magic script It’s just a tiny, obvious script, at least to me. :) BTW glad you got it going. I think Peter straightened out the with/as stuff. I’ll just say that the line that started out #withwas correct as written. The intent was that one would simply remove the comment (with ONE keypress) to enable the line. Note that my coding style is that if a line is an informational comment it starts out as # plus space plus text. If it is code that is commented out, it is # (at correct indent level), then no space(!), then code.[A-Z] person What is that? Why is it a lucky thing? 
- 
 Note that my coding style is that if a line is an informational comment it starts out as # plus space plus text. If it is code that is commented out, it is # (at correct indent level), then no space(!), then code very nice and clean to read 👍 [A-Z] person hahaha, i never thought that [A-Z] person could be interpreted as a potential insult, or discrimination, but it is not. 😂😂😂 it’s just an expression for languages without special characters and letters like áàñøö and so on. 
- 
 Hello @alan-kilborn, @meta-chuh and All, First, Alan, the expression “An [A-Z]person” is a common way, for @scott-sumner to point out that he’s poorly concerned about accentuated characters and all relative questions ! That’s why I said that you’re lucky for not having to bother about these problems ;-))Not also that I said, above, “is a common way” and not “was a common way” as I do hope that Scott will be back, on our forum, very soon ! 
 Now, of course, the Python syntax, below, is totally exact ! with open(r'sr_list.txt') as f: sr_list = f.readlines()It’s just that when I saw the two comment lines : # or take input from a file: #with open(r'sr_list.txt') as f: sr_list = f.readlines()I thought, wrongly, it meant, in fact : # or take input from a file with [ the sentence ]: # open(r'sr_list.txt') as f: sr_list = f.readlines()
 BTW, Alan, I tested, in the sr_list.txt, the syntax |^|Test, with some space chars after the word Test and, unfortunately, the ending spaces are not taken in account. Of course, I could have used|^|Test\x20\x20\x20…So, may I ask for two improvements : - 
The possibility to repeat the separator, after the replacement string, to take extra blank chars in account, either, in the sr_list.txtfile or in the script, itself
- 
The possibility to add comments, beginning with the usual #char, in thesr-list.txtfile
 For instance : # Add the string ABC, followed with 3 SPACES, at BEGINNING of line |^|ABC | # Add the string XYZ, followed with 3 SPACES at END of line !$!XYZ !
 Contrary to what I said, @meta-chuh, I didn’t come back and just preferred going to bed as I’ve planned to spend a ski-day, as weather was quite nice, Wednesday, on Grenoble and, in addition, I also met some friends of mine, in Chamrousse ski-station ;-)) As promised, I installed the last 1.3.0.0.Python script version in my local N++v7.6.2installationLet’s suppose that is N++ v7.6.2is installed in any folderXXXX, different from folderC\Program filesand folderC\Program files (x86). Then,- 
I downloaded the PythonScript_Full_1.3.0.0.ziparchive, inXXXXfolder
- 
With 7zFileManager, I extracted all archive’s contents, in the XXXXfolder
- 
I needed to execute an extra task : - Move the library PythonScript.dllfrom thepluginsfolder to theplugins\PythonScriptfolder
 
- Move the library 
- 
I opened Notepad v7.6.2- I chose the menu option Plugins > Python Script > New Scriptand, immediately closed the window, with the ESC key, in order to create the file treeXXXX > plugins > Config > PythonScript > scripts
 
- I chose the menu option 
 
 Finally, here is, below, the main file’s layout, right after installing the last Python script v1.3.0.0::XXXX, below, represents the INSTALL folder of N++ v7.6.2 , which must be DIFFERENT from, either, "C\Program files" and "C\Program files (x86)" It's IMPORTANT to note that this LOCAL installation needs the ZERO-LENGTH file, "doLocalConf.xml", along with "notepad..exe" XXXX \ |-- autoCompletion (folder) | \ | |-- ".xml" files | |-- localization (folder) | \ | |-- ".xml" files | |-- plugins (folder) | \ | |-- Config (folder) | | \ | | |-- Hunspell (folder) | | | \ | | | |-- en_US.aff | | | | | | | |-- en_US.dic | | | | | |-- PythonScript (folder) | | | \ | | | |-- scripts (folder) | | | \ | | | |-- Future USER ".py" scripts | | | | | |-- ".ini" files | | | | | |-- nppPluginList.dll | | | |-- doc (folder) | | \ | | |-- PythonScript(folder) | | \ | | |-- _sources (folder) | | | | | |-- _static (folder) | | | | | |-- ".html" files and Miscellaneous files | | | |-- DSpellCheck (folder) | | \ | | |-- DSpellCheck.dll | | | |-- mineTools (folder) | | \ | | |-- mineTools.dll | | | |-- NppConverter (folder) | | \ | | |-- NppConverter.dll | | | |-- NppExport | | \ | | |-- NppExport.dll | | | |-- PythonScript (folder) | | \ | | |-- lib (folder) | | | \ | | | |-- Sub-folders | | | | | | | |-- ".py" files | | | | | |-- scripts (folder) | | | \ | | | |-- Samples (folder) | | | | \ | | | | |-- ".py" scripts | | | | | | | |-- startup.py | | | | | |-- PythonScript.dll | |-- themes (folder) | \ | |-- ".xml" files | |-- updater (folder) | \ | |-- GUP.exe | | | |-- gup.xml | | | |-- libcurl.dll | |-- doLocalConf.xml | |-- Notepad++.exe | |-- python27.dll | |-- SciLexer.dll | |-- ".txt" files | |-- ".xml" CONFIGURATION filesBest Regards, guy038 P.S. : In the future, I think that, at least, for portable installs, when all problems concerning “Plugins Admin” are solved, it would be reasonable to migrate the Configanddocdirectories from thepluginsfolder to the higher level, with the other directorieslocalization,autoCompletion,themesandupdaterSo, the pluginsfolder would only contains sub-folders relative to each plugin installed ! What do you think of my idea ?
- 
- 
 Hello @alan-kilborn, @meta-chuh and All, I’m answering to myself, concerning the last question, at the end of my previous post Eventually, it would not be a nice solution to do so as, indeed, the Configanddocfolders contain, both, files rather relative to plugins, too !Cheers, guy038 
- 
 @guy038 said: may I ask for two improvements We don’t really need to repeat the delimiter, we just need to NOT ignore trailing space. What causes an ignoring of the trailing space in the original script is the rstrip()function. By default this function removes all whitespace from the right side of a string. If we change it to tell it to only strip line ending characters, it will leave blanks on that side:rstrip('\n'). Note that this will work for line endings of\nor\r\nin the file. I mention this because at first glance it would appear to only work for line endings of\nbut that is not the case.Using #as a comment character is also easy, we can do it with this logic:if line[0] == '#': continuewhich means "if the first column of the data is#then “continue” the “for” loop by jumping back up to the “for” line, ignoring the rest of the indented lines under the “for”.A new version of the “magic” (still LOL!) script is: # format for each line is: delimiter then search regex then delimiter then replace regex sr_list = [ '!a!A ', '# I start with # so I am merely a comment line', '@b@B', '!c!C', ] # or take input from a file: #with open(r'sr_list.txt') as f: sr_list = f.readlines() editor.beginUndoAction() for line in sr_list: if line[0] == '#': continue (s,r) = line[1:].rstrip('\n').split(line[0]) editor.rereplace(s,r) editor.endUndoAction()
- 
 Not also that I said, above, “is a common way” and not “was a common way” as I do hope that Scott will be back, on our forum, very soon ! me too, and i think all others too … a little secret: i saw him active at the npp github repo a few days ago 😃👍 … but don’t tell anyone ;-) I didn’t come back and just preferred going to bed as I’ve planned to spend a ski-day, as weather was quite,nice Wednesday, on Grenoble and, in addition, I also met some friends of mine, in Chamrousse ski-station ;-)) good done, best thing to do … but envyyyyyy ;-) XXXX ( INSTALL folder of N++ v7.6.2 , DIFFERENT from folder “C\Program files” and folder “C\Program files (x86)” ) thanks for your tree, it comes in very handy and i’ve bookmarked it. for competition i would edit it to: 
 XXXX ( PORTABLE folder of N++ v7.6.2 , DIFFERENT from folder "C\Program files" and folder "C\Program files (x86)" )
 and/or a note that doLocalConf.xml has to be present.
 just to make sure readers will not get those structures mixed up with the different folder structure of an installed version without doLocalConf.xml.
- 
 Hello, @v-s-rawat, @alan-kilborn, @meta-chuh and All, Alan, I tried your second version and everything went OK ! However, I prefer having a final separator, in order to easily see, in the SR_list.txt, the contents of the replacement regex. So, here is, below, my own version of your excellent script : #coding=utf-8 import re # -------------------------------------------------------------------------------------------------------------------------------------- # Script "Multiples_SR.py" # A LITTLE adaptation from an ORIGINAL and VALUABLE script of Alan KILBORN ( January 2019 ) ! # See https://notepad-plus-plus.org/community/topic/16942/pythonscript-any-ready-pyscript-to-replace-one-huge-set-of-regex-phrases-with-others/21 # This script : # - Reads an existing "SR_List.txt" file, of the CURRENT directory, containing a list of SEARCH/REPLACEMENT strings, ONE PER line # - Selects, one at a time, a COUPLE of SEARCH and REPLACEMENT regexes / expressions / strings / characters # - Executes this present S/R on CURRENT edited file, in NOTEPAD++ # - Loop till the END of file # Any PURE BLANK line or COMMENT line, beginning with '#', of the "SR_list.txt" file, are simply IGNORED # -------------------------------------------------------------------------------------------------------------------------------------- # For EACH line, in the "SR_List.txt" file, the format is <DELIMITER><SEARCH regex><DELIMITER><REPLACE regex><DELIMITER> ## EXAMPLES : ## ¯¯¯¯¯¯¯¯ ## Deletes any [ending] "; comment" / Delimiter = '!' #!(?-s)(^.*?);.+!\1! ## Changes any LOWER-case string "notepad++" in its UPPER-case equivalent / Delimiter = '@' #@(?-i)notepad\+\+@NOTEPAD++@ ## Changes any "Smith" and 'James' strings, with that EXACT case, to, respectively, "Name" and "First name" / Delimiter = '&' ## Deletes any "TEST" string, with that EXACT case #&(Smith)|TEST|(James)&(?1Name)(?2First name)& ## Replaces any BACKSLASH character with the "123" number, both preceded and followed with 3 SPACE characters / Delimiter = '%' #%\\% 123 % ## or, also, the syntax %\x5c% 123 % ## Deletes any string "Fix", followed with a SPACE char, whatever its CASE / Delimiter = '+' #+(?i)Fix ++ ## Change 3 CONSECUTIVE "#" characters with 3 BACKSLASH characters / Delimiter = '*' #*###*\\\\\\* # -------------------------------------------------------------------------------------------------------------------------------------- # In the CODE line, right below, you may : # - Modify the NAME of the file, containing the SEARCH and REPLACEMENT regexes # - Indicate an ABSOLUTE or RELATIVE path, before the filename with open(r'SR_list.txt') as f: sr_list = f.readlines() # You may, as well, insert the SEARCH and REPLACE regexes, directly, in THIS script : #sr_list = [ # '!(?-s)(^.*?);.+!\\1!', # '@(?-i)notepad\\+\\+@NOTEPAD++@', # '&(Smith)|TEST|(James)&(?1Name)(?2First name)&', # '%\\\\% 123 %', # # or the syntax '%\x5c\x5c% 123 %', # '+(?i)Fix ++', # '*###*\\\\\\\\\\\\*', # ] # The use of RAW strings r'.......' is also possible, in order to SIMPLIFY some regexes # Note that these RAW regexes are strictly IDENTICAL to those, which could be contained in a "SR_List.txt" file, WITHOUT the 'r' PREFIX #sr_list = [ # r'!(?-s)(^.*?);.+!\1!', # r'@(?-i)notepad\+\+@NOTEPAD++@', # r'&(Smith)|TEST|(James)&(?1Name)(?2First name)&', # r'%\\% 123 %', # # or the syntax r'%\x5c% 123 %', # r'+(?i)Fix ++', # r'*###*\\\\\\*', # ] editor.beginUndoAction() console.write ('\nMODIFICATIONS on FILE "{}: "\n\n'.format(notepad.getCurrentFilename())) # Note : Variable e is always EMPTY string ( Part AFTER the THIRD delimiter and BEFORE the END of line ! ) for line in sr_list: if line[0] == '#' or line == '\n' : continue (s,r,e) = line[1:].rstrip('\n').split(line[0]) console.write(' SEARCH : >{}<\n'.format(s)) console.write(' REPLACE : >{}<\n\n'.format(r)) editor.rereplace(s,r) # or editor.rereplace(s,r,re.IGNORECASE) / editor.rereplace(s,r,re.I) editor.endUndoAction() # END of Multiple_SR.py script
 @meta-Chuh, as you said, I slightly modify the local Notepad++ tree, in my previous post, to point out the importance of the doLocalConf.xmlfile ;-))
 Cheers,guy038 



