@@ -269,7 +269,8 @@ void MultiReplace::positionAndResizeControls(int windowWidth, int windowHeight)
269
269
ctrlMap[IDC_MATCH_CASE_CHECKBOX] = { sx (16 ), sy (101 ), sx (158 ), checkboxHeight, WC_BUTTON, getLangStrLPCWSTR (L" panel_match_case" ), BS_AUTOCHECKBOX | WS_TABSTOP, NULL };
270
270
ctrlMap[IDC_USE_VARIABLES_CHECKBOX] = { sx (16 ), sy (126 ), sx (134 ), checkboxHeight, WC_BUTTON, getLangStrLPCWSTR (L" panel_use_variables" ), BS_AUTOCHECKBOX | WS_TABSTOP, NULL };
271
271
ctrlMap[IDC_USE_VARIABLES_HELP] = { sx (152 ), sy (126 ), sx (20 ), sy (20 ), WC_BUTTON, getLangStrLPCWSTR (L" panel_help" ), BS_PUSHBUTTON | WS_TABSTOP, NULL };
272
- ctrlMap[IDC_REPLACE_FIRST_CHECKBOX] = { sx (16 ), sy (151 ), sx (158 ), checkboxHeight, WC_BUTTON, getLangStrLPCWSTR (L" panel_replace_first_match_only" ), BS_AUTOCHECKBOX | WS_TABSTOP, NULL };
272
+ ctrlMap[IDC_REPLACE_FIRST_CHECKBOX] = { sx (16 ), sy (151 ), sx (112 ), checkboxHeight, WC_BUTTON, getLangStrLPCWSTR (L" panel_replace_first_match_only" ), BS_AUTOCHECKBOX | WS_TABSTOP, NULL };
273
+ ctrlMap[IDC_REPLACE_HIT_EDIT] = { sx (130 ), sy (151 ), sx (41 ), sy (16 ), WC_EDIT, NULL , ES_LEFT | WS_BORDER | WS_TABSTOP | ES_AUTOHSCROLL, NULL };
273
274
ctrlMap[IDC_WRAP_AROUND_CHECKBOX] = { sx (16 ), sy (176 ), sx (158 ), checkboxHeight, WC_BUTTON, getLangStrLPCWSTR (L" panel_wrap_around" ), BS_AUTOCHECKBOX | WS_TABSTOP, NULL };
274
275
275
276
ctrlMap[IDC_SEARCH_MODE_GROUP] = { sx (180 ), sy (79 ), sx (173 ), sy (104 ), WC_BUTTON, getLangStrLPCWSTR (L" panel_search_mode" ), BS_GROUPBOX, NULL };
@@ -4232,14 +4233,30 @@ bool MultiReplace::replaceAll(const ReplaceItemData& itemData, int& findCount, i
4232
4233
context.docLength = send (SCI_GETLENGTH, 0 , 0 );
4233
4234
context.isColumnMode = (IsDlgButtonChecked (_hSelf, IDC_COLUMN_MODE_RADIO) == BST_CHECKED);
4234
4235
context.isSelectionMode = (IsDlgButtonChecked (_hSelf, IDC_SELECTION_RADIO) == BST_CHECKED);
4235
- context.retrieveFoundText = itemData.useVariables ;;
4236
+ context.retrieveFoundText = itemData.useVariables ;
4236
4237
context.highlightMatch = false ;
4237
4238
4238
4239
send (SCI_SETSEARCHFLAGS, context.searchFlags );
4239
4240
4240
4241
SearchResult searchResult = performSearchForward (context, 0 );
4241
4242
4242
4243
bool isReplaceFirstEnabled = (IsDlgButtonChecked (_hSelf, IDC_REPLACE_FIRST_CHECKBOX) == BST_CHECKED);
4244
+
4245
+ std::vector<int > selectedMatches;
4246
+
4247
+ // Only process the match selection if the option is enabled
4248
+ if (isReplaceFirstEnabled) {
4249
+ selectedMatches = parseNumberRanges (
4250
+ getTextFromDialogItem (_hSelf, IDC_REPLACE_HIT_EDIT),
4251
+ getLangStr (L" status_invalid_match_selection" )
4252
+ );
4253
+
4254
+ // If the Edit field is empty, default to replacing only the first match
4255
+ if (selectedMatches.empty ()) {
4256
+ selectedMatches.push_back (1 );
4257
+ }
4258
+ }
4259
+
4243
4260
int previousLineIndex = -1 ;
4244
4261
int lineFindCount = 0 ;
4245
4262
@@ -4307,30 +4324,34 @@ bool MultiReplace::replaceAll(const ReplaceItemData& itemData, int& findCount, i
4307
4324
}
4308
4325
4309
4326
Sci_Position newPos;
4310
- if (!skipReplace) {
4311
- if (itemData.regex ) {
4312
- newPos = performRegexReplace (replaceTextUtf8, searchResult.pos , searchResult.length );
4327
+
4328
+ // ** Check if the current match should be replaced **
4329
+ if (!isReplaceFirstEnabled || std::find (selectedMatches.begin (), selectedMatches.end (), findCount) != selectedMatches.end ())
4330
+ {
4331
+ if (!skipReplace) {
4332
+ if (itemData.regex ) {
4333
+ newPos = performRegexReplace (replaceTextUtf8, searchResult.pos , searchResult.length );
4334
+ }
4335
+ else {
4336
+ newPos = performReplace (replaceTextUtf8, searchResult.pos , searchResult.length );
4337
+ }
4338
+ replaceCount++;
4339
+
4340
+ if (itemIndex != SIZE_MAX) { // check if used in List
4341
+ updateCountColumns (itemIndex, -1 , replaceCount);
4342
+ }
4343
+
4344
+ context.docLength = send (SCI_GETLENGTH, 0 , 0 );
4313
4345
}
4314
4346
else {
4315
- newPos = performReplace (replaceTextUtf8, searchResult.pos , searchResult.length );
4347
+ newPos = searchResult.pos + searchResult.length ;
4348
+ // Clear selection
4349
+ send (SCI_SETSELECTIONSTART, newPos, 0 );
4350
+ send (SCI_SETSELECTIONEND, newPos, 0 );
4316
4351
}
4317
- replaceCount++;
4318
-
4319
- if (itemIndex != SIZE_MAX) { // check if used in List
4320
- updateCountColumns (itemIndex, -1 , replaceCount);
4321
- }
4322
-
4323
- context.docLength = send (SCI_GETLENGTH, 0 , 0 );
4324
4352
}
4325
4353
else {
4326
- newPos = searchResult.pos + searchResult.length ;
4327
- // Clear selection
4328
- send (SCI_SETSELECTIONSTART, newPos, 0 );
4329
- send (SCI_SETSELECTIONEND, newPos, 0 );
4330
- }
4331
-
4332
- if (isReplaceFirstEnabled) {
4333
- break ; // Exit the loop after the first successful replacement
4354
+ newPos = searchResult.pos + searchResult.length ; // Move to next match without replacing
4334
4355
}
4335
4356
4336
4357
searchResult = performSearchForward (context, newPos);
@@ -6598,66 +6619,26 @@ bool MultiReplace::parseColumnAndDelimiterData() {
6598
6619
std::string extendedDelimiter = convertAndExtend (delimiterData, true );
6599
6620
std::string quoteCharConverted = wstringToString (quoteCharString);
6600
6621
6601
- // ** Check for changes BEFORE modifying existing values**
6622
+ // Check for changes BEFORE modifying existing values
6602
6623
bool delimiterChanged = (columnDelimiterData.extendedDelimiter != extendedDelimiter);
6603
6624
bool quoteCharChanged = (columnDelimiterData.quoteChar != quoteCharConverted);
6604
6625
6605
- // **Process column data before clearing old values**
6606
- std::set<int > parsedColumns;
6607
- std::vector<int > parsedInputColumns;
6608
-
6609
6626
// Trim leading and trailing commas from column data
6610
6627
columnDataString.erase (0 , columnDataString.find_first_not_of (L' ,' ));
6611
6628
columnDataString.erase (columnDataString.find_last_not_of (L' ,' ) + 1 );
6612
6629
6630
+ // Ensure that columnDataString and delimiter are not empty
6613
6631
if (columnDataString.empty () || delimiterData.empty ()) {
6614
6632
showStatusMessage (getLangStr (L" status_missing_column_or_delimiter_data" ), COLOR_ERROR);
6615
6633
return false ;
6616
6634
}
6617
6635
6618
- // Lambda function to process a token (either a single column or a range like "1-3")
6619
- auto processToken = [&](const std::wstring& token) -> bool {
6620
- if (token.empty ()) return true ; // Ignore empty tokens
6621
- try {
6622
- size_t dashPos = token.find (L' -' );
6623
- if (dashPos != std::wstring::npos) { // Token represents a range (e.g., "1-3")
6624
- int startRange = std::stoi (token.substr (0 , dashPos));
6625
- int endRange = std::stoi (token.substr (dashPos + 1 ));
6626
- if (startRange < 1 || endRange < startRange) {
6627
- showStatusMessage (getLangStr (L" status_invalid_range_in_column_data" ), COLOR_ERROR);
6628
- return false ;
6629
- }
6630
- for (int i = startRange; i <= endRange; ++i) {
6631
- if (parsedColumns.insert (i).second ) {
6632
- parsedInputColumns.push_back (i);
6633
- }
6634
- }
6635
- }
6636
- else { // Token represents a single column (e.g., "5")
6637
- int col = std::stoi (token);
6638
- if (col < 1 ) {
6639
- showStatusMessage (getLangStr (L" status_invalid_column_number" ), COLOR_ERROR);
6640
- return false ;
6641
- }
6642
- if (parsedColumns.insert (col).second ) {
6643
- parsedInputColumns.push_back (col);
6644
- }
6645
- }
6646
- }
6647
- catch (const std::exception &) {
6648
- showStatusMessage (getLangStr (L" status_syntax_error_in_column_data" ), COLOR_ERROR);
6649
- return false ;
6650
- }
6651
- return true ;
6652
- };
6636
+ // Use parseNumberRanges() to process column data
6637
+ std::vector<int > parsedColumns = parseNumberRanges (columnDataString, getLangStr (L" status_invalid_range_in_column_data" ));
6638
+ if (parsedColumns.empty ()) return false ; // Abort if parsing failed
6653
6639
6654
- // Tokenize columnDataString using a stream and process each token
6655
- std::wistringstream iss (columnDataString);
6656
- std::wstring token;
6657
- while (std::getline (iss, token, L' ,' )) {
6658
- if (!processToken (token))
6659
- return false ;
6660
- }
6640
+ // Convert parsedColumns to set for uniqueness
6641
+ std::set<int > uniqueColumns (parsedColumns.begin (), parsedColumns.end ());
6661
6642
6662
6643
// Validate delimiter
6663
6644
if (extendedDelimiter.empty ()) {
@@ -6672,16 +6653,16 @@ bool MultiReplace::parseColumnAndDelimiterData() {
6672
6653
return false ;
6673
6654
}
6674
6655
6675
- // ** Check if columns have changed BEFORE updating stored data**
6676
- bool columnChanged = (columnDelimiterData.columns != parsedColumns );
6656
+ // Check if columns have changed BEFORE updating stored data
6657
+ bool columnChanged = (columnDelimiterData.columns != uniqueColumns );
6677
6658
6678
6659
// Update columnDelimiterData values
6679
6660
columnDelimiterData.delimiterChanged = delimiterChanged;
6680
6661
columnDelimiterData.quoteCharChanged = quoteCharChanged;
6681
6662
columnDelimiterData.columnChanged = columnChanged;
6682
6663
6683
- columnDelimiterData.inputColumns = std::move (parsedInputColumns );
6684
- columnDelimiterData.columns = std::move (parsedColumns );
6664
+ columnDelimiterData.inputColumns = std::move (parsedColumns );
6665
+ columnDelimiterData.columns = std::move (uniqueColumns );
6685
6666
columnDelimiterData.extendedDelimiter = std::move (extendedDelimiter);
6686
6667
columnDelimiterData.delimiterLength = columnDelimiterData.extendedDelimiter .length ();
6687
6668
columnDelimiterData.quoteChar = std::move (quoteCharConverted);
@@ -7920,6 +7901,70 @@ int MultiReplace::getFontHeight(HWND hwnd, HFONT hFont) {
7920
7901
return fontHeight; // Return the font height
7921
7902
}
7922
7903
7904
+ std::vector<int > MultiReplace::parseNumberRanges (const std::wstring& input, const std::wstring& errorMessage)
7905
+ {
7906
+ std::vector<int > result;
7907
+ if (input.empty ()) return result; // Return empty vector if input is empty
7908
+
7909
+ std::set<int > uniqueNumbers;
7910
+ std::wistringstream stream (input);
7911
+ std::wstring token;
7912
+
7913
+ // Lambda function to process each token (either a single number or a range "1-5")
7914
+ auto processToken = [&](const std::wstring& token) -> bool
7915
+ {
7916
+ if (token.empty ()) return true ; // Ignore empty tokens
7917
+
7918
+ try
7919
+ {
7920
+ size_t dashPos = token.find (L' -' );
7921
+ if (dashPos != std::wstring::npos)
7922
+ {
7923
+ // Extract range start and end
7924
+ int startRange = std::stoi (token.substr (0 , dashPos));
7925
+ int endRange = std::stoi (token.substr (dashPos + 1 ));
7926
+
7927
+ // Validate range
7928
+ if (startRange < 1 || endRange < startRange)
7929
+ return false ;
7930
+
7931
+ // Insert numbers in the range
7932
+ for (int i = startRange; i <= endRange; ++i)
7933
+ uniqueNumbers.insert (i);
7934
+ }
7935
+ else
7936
+ {
7937
+ // Single number
7938
+ int number = std::stoi (token);
7939
+ if (number < 1 )
7940
+ return false ; // Invalid input
7941
+
7942
+ uniqueNumbers.insert (number);
7943
+ }
7944
+ }
7945
+ catch (const std::exception &)
7946
+ {
7947
+ return false ; // Invalid input format
7948
+ }
7949
+
7950
+ return true ;
7951
+ };
7952
+
7953
+ // Split input by comma and process each token
7954
+ while (std::getline (stream, token, L' ,' ))
7955
+ {
7956
+ if (!processToken (token))
7957
+ {
7958
+ showStatusMessage (errorMessage, COLOR_ERROR);
7959
+ return {}; // Invalid input -> return empty vector
7960
+ }
7961
+ }
7962
+
7963
+ result.assign (uniqueNumbers.begin (), uniqueNumbers.end ());
7964
+ return result;
7965
+ }
7966
+
7967
+
7923
7968
#pragma endregion
7924
7969
7925
7970
0 commit comments