Community
    • Login

    TextFX doesn't work

    Scheduled Pinned Locked Moved Help wanted · · · – – – · · ·
    12 Posts 3 Posters 5.4k Views
    Loading More Posts
    • Oldest to Newest
    • Newest to Oldest
    • Most Votes
    Reply
    • Reply as topic
    Log in to reply
    This topic has been deleted. Only users with topic management privileges can see it.
    • Scott SumnerS
      Scott Sumner @LAPIII
      last edited by

      @LAPIII

      I’m not at all a big fan of TextFX, but it seemed to work for what I guessed might resemble your situation (see below for my “before” and “after”). Maybe if you’d have shown some sample data where it doesn’t work for you…

      Imgur

      1 Reply Last reply Reply Quote 4
      • Scott SumnerS
        Scott Sumner
        last edited by

        This isn’t the best code I’ve ever written [ :-D ] but it seems to mimic the TextFX functionality, and should run on 64-bit where TextFX isn’t–and likely will never be–available.

        It requires the Pythonscript plugin; I call it LineUpMultipleLinesBySpecifiedCharacter.py:

        def LUMLBSC_main():
        
            ps_name = 'Line Up Multiple Lines By Specified Character'
        
            line_ending = ['\r\n', '\r', '\n'][notepad.getFormatType()]
        
            (first_line, last_line) = editor.getUserLineSelection()
            if first_line == 0 and last_line == editor.getLineCount() - 1:
                if notepad.messageBox('\r\n'.join([
                    'Do you want to line up ALL lines in the file by the alignment character',
                    '(which you will specify in a moment)',
                    '?',
                    '\r\n',
                    'If not, create a selection touching lines to affect, and re-run...',
                    ]),
                    ps_name, MESSAGEBOXFLAGS.YESNO) != MESSAGEBOXFLAGS.RESULTYES: return
        
            result = notepad.prompt('Enter delimiter:', ps_name, ',')
            if result == None or len(result) == 0: return
            delimiter = result
        
            line_interdelim_list_of_lists = []
            longest_length_list = []
        
            (selection_start, selection_end) = editor.getUserCharSelection()
            start_position_selected_lines = editor.positionFromLine(editor.lineFromPosition(selection_start))
            end_position_selected_lines = editor.getLineEndPosition(editor.lineFromPosition(selection_end))
        
            lines = editor.getTextRange(start_position_selected_lines, end_position_selected_lines).splitlines()
        
            # find the longest interdelimiter piece between each delimiter:
            for line in lines:
                interdelimiter_list = line.split(delimiter)
                line_interdelim_list_of_lists.append(interdelimiter_list)
                num_delimiters_this_line = len(interdelimiter_list) - 1
                if num_delimiters_this_line > 0:
                    if len(longest_length_list) < num_delimiters_this_line + 1:
                        longest_length_list += [0] * (num_delimiters_this_line + 1 - len(longest_length_list))
                    for (j, interdelimiter_string) in enumerate(interdelimiter_list):
                        if len(interdelimiter_string) > longest_length_list[j]: longest_length_list[j] = len(interdelimiter_string)
        
            # build up new lines:
            new_lines_list = []
            for old_interdelimiter_list in line_interdelim_list_of_lists:
                if len(old_interdelimiter_list) - 1 > 0:
                    new_interdelimiter_list = []
                    for (j, interdelimiter_string) in enumerate(old_interdelimiter_list):
                        if len(interdelimiter_string) < longest_length_list[j] and j != len(old_interdelimiter_list) - 1:
                            new_interdelimiter_list.append(interdelimiter_string + ' ' * (longest_length_list[j] - len(interdelimiter_string)))
                        else:
                            new_interdelimiter_list.append(interdelimiter_string)
                    line = delimiter.join(new_interdelimiter_list)
                else:
                    line = delimiter.join(old_interdelimiter_list)
                new_lines_list.append(line)
        
            # replace original lines with new lines:
            editor.setTarget(start_position_selected_lines, end_position_selected_lines)
            editor.replaceTarget(line_ending.join(new_lines_list))
        
        LUMLBSC_main()
        
        1 Reply Last reply Reply Quote 1
        • guy038G
          guy038
          last edited by

          Hello, @scott-sumner and All ,

          I’ve just tried your script and it’s working nice ! Now, if you copy/paste the following blocks of text in Notepad++, unset the Word wrap option, for best reading !

          Two remarks :


          A)

          • From your initial text, below :
          Hot, with food story told once chief seven, press smile often, modern begin did.
          Sugar town gold heart boat modern dead, did; meet crowd tube several figure.
          Four; loud brought our before size segment south; open second, bright answer board stood; over.
          Offer him sentence caught necessary feet just plan write noise rope question foot.
          Can early young stand, tie most expect play value black!
          Pick thick special, left hair thank grass boat; morning.
          Him corn, look cotton they match floor fair came also better side stretch quite.
          Sent now favor rock type tail huge; good flat where opposite try add.
          Wrong kill use point may broad region grand king minute.
          Term organ value, million ring charge view, control map, sight wall, tool interest.
          

          After running the script, we get the following text :

          Hot                                                          , with food story told once chief seven                                 , press smile often, modern begin did.
          Sugar town gold heart boat modern dead                       , did; meet crowd tube several figure.
          Four; loud brought our before size segment south; open second, bright answer board stood; over.
          Offer him sentence caught necessary feet just plan write noise rope question foot.
          Can early young stand                                        , tie most expect play value black!
          Pick thick special                                           , left hair thank grass boat; morning.
          Him corn                                                     , look cotton they match floor fair came also better side stretch quite.
          Sent now favor rock type tail huge; good flat where opposite try add.
          Wrong kill use point may broad region grand king minute.
          Term organ value                                             , million ring charge view                                              , control map      , sight wall       , tool interest.
          

          I noticed that the last comma sign ( the 4th ), of the last line, seems aligned with a virtual comma char, located after the end of the first line. So, to be logic, it would be better to consider that the first comma of every line, should begin AFTER the longest line of text, which has not a comma character, at all ! ( the 4th line, in your example )

          So, we would obtain :

          Hot                                                                               , with food story told once chief seven                                 , press smile often, modern begin did.
          Sugar town gold heart boat modern dead                                            , did; meet crowd tube several figure.
          Four; loud brought our before size segment south; open second                     , bright answer board stood; over.
          Offer him sentence caught necessary feet just plan write noise rope question foot.
          Can early young stand                                                             , tie most expect play value black!
          Pick thick special                                                                , left hair thank grass boat; morning.
          Him corn                                                                          , look cotton they match floor fair came also better side stretch quite.
          Sent now favor rock type tail huge; good flat where opposite try add.
          Wrong kill use point may broad region grand king minute.
          Term organ value                                                                  , million ring charge view                                              , control map      , sight wall       , tool interest.
          

          It look easier to read ;-))


          B)

          May be, a second version, of your script, could take delimiters in account ! I mean that the specified character would not be considered if found between a list of specific delimiters, or after some comment delimiters until the end of current line !

          Of course, it’s just some thoughts and, simply, do as you like !

          Best Regards,

          guy038

          Scott SumnerS 1 Reply Last reply Reply Quote 2
          • Scott SumnerS
            Scott Sumner @guy038
            last edited by

            Hello @guy038 ,

            Thank you for taking an interest in the script. My purpose was to exactly duplicate the TextFX functionality, but after reading your comments and some further testing, I see that it doesn’t quite do that, so I offer up a revised version, below.

            Regarding enhancements over and above the TextFX functionality, let’s not go there with this one. I am, however, working up something in response to this thread which will maybe meet a more flexible alignment need (and it probably will involve regular expressions!)…

            def LUMLBSC_main():
            
                ps_name = 'Line Up Multiple Lines By Specified Character'
            
                line_ending = ['\r\n', '\r', '\n'][notepad.getFormatType()]
            
                (first_line, last_line) = editor.getUserLineSelection()
                if first_line == 0 and last_line == editor.getLineCount() - 1:
                    if notepad.messageBox('\r\n'.join([
                        'Do you want to line up ALL lines in the file by the alignment character',
                        '(which you will specify in a moment)',
                        '?',
                        '\r\n',
                        'If not, create a selection touching lines to affect, and re-run...',
                        ]),
                        ps_name, MESSAGEBOXFLAGS.YESNO) != MESSAGEBOXFLAGS.RESULTYES: return
            
                result = notepad.prompt('Enter delimiter:', ps_name, ',')
                if result == None or len(result) == 0: return
                delimiter = result
            
                line_interdelim_list_of_lists = []
                longest_length_list = []
            
                (selection_start, selection_end) = editor.getUserCharSelection()
                start_position_selected_lines = editor.positionFromLine(editor.lineFromPosition(selection_start))
                end_position_selected_lines = editor.getLineEndPosition(editor.lineFromPosition(selection_end))
            
                lines = editor.getTextRange(start_position_selected_lines, end_position_selected_lines).splitlines()
            
                # find the longest interdelimiter piece between each delimiter:
                for line in lines:
                    interdelimiter_list = line.split(delimiter)
                    line_interdelim_list_of_lists.append(interdelimiter_list)
                    num_delimiters_this_line = len(interdelimiter_list) - 1
                    if num_delimiters_this_line > 0:
                        if len(longest_length_list) < num_delimiters_this_line: longest_length_list += [0] * (num_delimiters_this_line - len(longest_length_list))
                        for (j, interdelimiter_string) in enumerate(interdelimiter_list):
                            if j != len(interdelimiter_list) - 1:
                                if len(interdelimiter_string) > longest_length_list[j]: longest_length_list[j] = len(interdelimiter_string)
            
                # build up new lines:
                new_lines_list = []
                for old_interdelimiter_list in line_interdelim_list_of_lists:
                    if len(old_interdelimiter_list) - 1 > 0:
                        new_interdelimiter_list = []
                        for (j, interdelimiter_string) in enumerate(old_interdelimiter_list):
                            if (j != len(old_interdelimiter_list) - 1) and (len(interdelimiter_string) < longest_length_list[j]):
                                new_interdelimiter_list.append(interdelimiter_string + ' ' * (longest_length_list[j] - len(interdelimiter_string)))
                            else:
                                new_interdelimiter_list.append(interdelimiter_string)
                        line = delimiter.join(new_interdelimiter_list)
                    else:
                        line = delimiter.join(old_interdelimiter_list)
                    new_lines_list.append(line)
            
                # replace original lines with new lines:
                editor.setTarget(start_position_selected_lines, end_position_selected_lines)
                editor.replaceTarget(line_ending.join(new_lines_list))
            
            LUMLBSC_main()
            
            1 Reply Last reply Reply Quote 1
            • guy038G
              guy038
              last edited by guy038

              Hello, @scott-sumner and All ,

              OK, got it ! Changes concern the first and last line of your original test. More logical, of course !


              Now, Scott, I found an easy work-around to get my “preferred line-up” behavior ! ( Rather stubborn, the guy ! )

              • Firstly, I use the regex S/R : SEARCH (?-s)^(?!.*,).+ and REPLACE $0,, which adds a comma at the end of all the lines, which do not contain any comma, yet.

              • Secondly, I run your LineUpMultipleLinesBySpecifiedCharacter.py script.

              • Thirdly, I use a second regex S/R : SEARCH \x20*,$ and REPLACE Leave EMPTY, which deletes any comma at the end of line, and its possible leading space characters.

              Et voilà,

              Cheers,

              guy038

              P.S. :

              Of course, in case of a list, made up some comma separated columns, it’s, likely, does not have a row, without any comma !

              1 Reply Last reply Reply Quote 2
              • Scott SumnerS
                Scott Sumner
                last edited by Scott Sumner

                One thing that I failed to mention earlier is that it doesn’t take much data for the TextFX version of this function to fail big-time:

                Imgur

                The Pythonscript version, however, easily handled a 1000-line file with average line length of 150 characters and a variable amount of commas per line. I cite these numbers–which aren’t really all that “large”–as that is something I tested it on; I wasn’t looking for any kind of “maximum capability” during my experimentation, just (hopefully) correct behavior.

                To get TextFX to handle the same file, I found I had to cut it down to around 40 lines so that it would not choke with the “Out of memory” error.

                1 Reply Last reply Reply Quote 1
                • guy038G
                  guy038
                  last edited by guy038

                  Hi, @Scott-sumner and All,

                  I found a small bug in your LineUpMultipleLinesBySpecifiedCharacter.py script, when text contains accentuated characters !

                  For instance, assuming the 3-lines text, below, containing some French words :

                  été,automobile,écart,table
                  entrepôt,avion,bien-être,couteau
                  parlementaire,tu,prestidigitateur,fourchette
                  

                  Running the present script, I obtained :

                  été        ,automobile,écart          ,table
                  entrepôt    ,avion     ,bien-être      ,couteau
                  parlementaire,tu        ,prestidigitateur,fourchette
                  

                  Of course, I would expect this nicer text, below :-))

                  été          ,automobile,écart           ,table
                  entrepôt     ,avion     ,bien-être       ,couteau
                  parlementaire,tu        ,prestidigitateur,fourchette
                  

                  Cheers

                  guy038

                  P.S. :

                  Scott, I’m just thinking about a nice improvement :

                  Could it be possible to define, in addition to the character to line up,( default comma character ), a string which would separate the different columns ?

                  For example :

                  • Any , would be changed into two space + , + two space chars

                  =>

                  été            ,  automobile  ,  écart             ,  table
                  entrepôt       ,  avion       ,  bien-être         ,  couteau
                  parlementaire  ,  tu          ,  prestidigitateur  ,  fourchette
                  
                  • Any , would be changed into two space + | + two space chars

                  =>

                  été            |  automobile  |  écart             |  table
                  entrepôt       |  avion       |  bien-être         |  couteau
                  parlementaire  |  tu          |  prestidigitateur  |  fourchette
                  
                  • Any , would be changed into the ; punctuation char + 6 space chars + the ; punctuation character, again

                  =>

                  été          ;      ;automobile;      ;écart           ;      ;table
                  entrepôt     ;      ;avion     ;      ;bien-être       ;      ;couteau
                  parlementaire;      ;tu        ;      ;prestidigitateur;      ;fourchette
                  
                  Scott SumnerS 1 Reply Last reply Reply Quote 1
                  • Scott SumnerS
                    Scott Sumner @guy038
                    last edited by

                    @guy038 said:

                    small bug…when text contains accentuated characters !

                    Hi Guy,

                    Don’t you know by now that I am an A-Z (and, correspondingly, a-z) sort of person? :-D

                    I don’t know…I wanted to confine this script to just mimic the TextFX functionality…on an (?i)[A-Z] basis, though, of course, heheheh…

                    I’m just thinking about a nice improvement…

                    You’ve seen enough Pythonscript by now…how about trying your hand at these modifications? :-)

                    1 Reply Last reply Reply Quote 1
                    • guy038G
                      guy038
                      last edited by guy038

                      Hi, @Scott-sumner and All,

                      OK, boss, done ! So, here is, below, the second version of the Scott script, that I called LineUpMultipleLinesBySpecifiedCharacter_v2.py. However, just note that is, merely, a simple re-copy of Scott’s code :-(

                      def LUMLBSC_v2_main():
                      
                          ps_name = 'Line Up Multiple Lines By Specified Character'
                      
                          line_ending = ['\r\n', '\r', '\n'][notepad.getFormatType()]
                      
                          (first_line, last_line) = editor.getUserLineSelection()
                          if first_line == 0 and last_line == editor.getLineCount() - 1:
                              if notepad.messageBox('\r\n'.join([
                                  'Do you want to line up ALL lines in the file by the alignment character',
                                  '(which you will specify in a moment)',
                                  '?',
                                  '\r\n',
                                  'If not, create a selection touching lines to affect, and re-run...',
                                  ]),
                                  ps_name, MESSAGEBOXFLAGS.YESNO) != MESSAGEBOXFLAGS.RESULTYES: return
                      
                          result = notepad.prompt('Enter delimiter:', ps_name, ',')
                          if result == None or len(result) == 0: return
                          delimiter = result
                      
                          separator = notepad.prompt('Enter the separator string, between columns:', ps_name, ',')
                          if separator == None or len(result) == 0: return
                      
                          line_interdelim_list_of_lists = []
                          longest_length_list = []
                      
                          (selection_start, selection_end) = editor.getUserCharSelection()
                          start_position_selected_lines = editor.positionFromLine(editor.lineFromPosition(selection_start))
                          end_position_selected_lines = editor.getLineEndPosition(editor.lineFromPosition(selection_end))
                      
                          lines = editor.getTextRange(start_position_selected_lines, end_position_selected_lines).splitlines()
                      
                          # find the longest interdelimiter piece between each delimiter:
                          for line in lines:
                              interdelimiter_list = line.split(delimiter)
                              line_interdelim_list_of_lists.append(interdelimiter_list)
                              num_delimiters_this_line = len(interdelimiter_list) - 1
                              if num_delimiters_this_line > 0:
                                  if len(longest_length_list) < num_delimiters_this_line: longest_length_list += [0] * (num_delimiters_this_line - len(longest_length_list))
                                  for (j, interdelimiter_string) in enumerate(interdelimiter_list):
                                      if j != len(interdelimiter_list) - 1:
                                          if len(interdelimiter_string) > longest_length_list[j]: longest_length_list[j] = len(interdelimiter_string)
                      
                          # build up new lines:
                          new_lines_list = []
                          for old_interdelimiter_list in line_interdelim_list_of_lists:
                              if len(old_interdelimiter_list) - 1 > 0:
                                  new_interdelimiter_list = []
                                  for (j, interdelimiter_string) in enumerate(old_interdelimiter_list):
                                      if (j != len(old_interdelimiter_list) - 1) and (len(interdelimiter_string) < longest_length_list[j]):
                                          new_interdelimiter_list.append(interdelimiter_string + ' ' * (longest_length_list[j] - len(interdelimiter_string)))
                                      else:
                                          new_interdelimiter_list.append(interdelimiter_string)
                                  line = separator.join(new_interdelimiter_list)
                              else:
                                  line = separator.join(old_interdelimiter_list)
                              new_lines_list.append(line)
                      
                          # replace original lines with new lines:
                          editor.setTarget(start_position_selected_lines, end_position_selected_lines)
                          editor.replaceTarget(line_ending.join(new_lines_list))
                      
                      LUMLBSC_v2_main()
                      

                      Strangely, if I chose an empty separator string, I thought that the script would exit without doing nothing. but I did get a text, without any separator

                      So, from this sample text ( which is the exact translation of my previous French text ! ) :

                      summer,automobile,space,table
                      warehouse,plane,wellbeing,knife
                      member of parliament,you,conjurer,fork
                      

                      It gives that text, without any separator between columns !

                      summer              automobilespace    table
                      warehouse           plane     wellbeingknife
                      member of parliamentyou       conjurer fork
                      

                      BR

                      guy038

                      P.S. :

                      But, I’m still short of ideas, regarding the accentuated characters ;-))

                      Scott SumnerS 1 Reply Last reply Reply Quote 2
                      • Scott SumnerS
                        Scott Sumner @guy038
                        last edited by

                        @guy038 said:

                        merely, a simple re-copy of Scott’s code

                        Wha? You made no changes except to add v2 in the name??

                        1 Reply Last reply Reply Quote 1
                        • guy038G
                          guy038
                          last edited by guy038

                          Hi, @Scott-sumner

                          I do not understand what you mean ?

                          I did added the part :

                              separator = notepad.prompt('Enter the separator string, between columns:', ps_name, ',')
                              if separator == None or len(result) == 0: return
                          

                          and I changed the lines :

                                      line = delimiter.join(new_interdelimiter_list)
                                  else:
                                      line = delimiter.join(old_interdelimiter_list)
                          

                          as :

                                      line = separator.join(new_interdelimiter_list)
                                  else:
                                      line = separator.join(old_interdelimiter_list)
                          

                          Not a great task, I agree, but, however, significant… and it works nice ;-))

                          BR

                          guy038

                          1 Reply Last reply Reply Quote 2
                          • First post
                            Last post
                          The Community of users of the Notepad++ text editor.
                          Powered by NodeBB | Contributors