diff --git a/.forceignore b/.forceignore index 02b2d9c2328..860395bcb93 100644 --- a/.forceignore +++ b/.forceignore @@ -4,4 +4,7 @@ # LWC Jest **/__tests__/** -**/__mocks__/** \ No newline at end of file +**/__mocks__/** +**/tsconfig.json + +**/*.ts diff --git a/CODEOWNERS b/CODEOWNERS new file mode 100644 index 00000000000..ce7a494556d --- /dev/null +++ b/CODEOWNERS @@ -0,0 +1,2 @@ +# Comment line immediately above ownership line is reserved for related other information. Please be careful while editing. +#ECCN:Open Source diff --git a/CODE_OF_CONDUCT.md b/CODE_OF_CONDUCT.md new file mode 100644 index 00000000000..b4612a7bc59 --- /dev/null +++ b/CODE_OF_CONDUCT.md @@ -0,0 +1,105 @@ +# Salesforce Open Source Community Code of Conduct + +## About the Code of Conduct + +Equality is a core value at Salesforce. We believe a diverse and inclusive +community fosters innovation and creativity, and are committed to building a +culture where everyone feels included. + +Salesforce open-source projects are committed to providing a friendly, safe, and +welcoming environment for all, regardless of gender identity and expression, +sexual orientation, disability, physical appearance, body size, ethnicity, nationality, +race, age, religion, level of experience, education, socioeconomic status, or +other similar personal characteristics. + +The goal of this code of conduct is to specify a baseline standard of behavior so +that people with different social values and communication styles can work +together effectively, productively, and respectfully in our open source community. +It also establishes a mechanism for reporting issues and resolving conflicts. + +All questions and reports of abusive, harassing, or otherwise unacceptable behavior +in a Salesforce open-source project may be reported by contacting the Salesforce +Open Source Conduct Committee at ossconduct@salesforce.com. + +## Our Pledge + +In the interest of fostering an open and welcoming environment, we as +contributors and maintainers pledge to making participation in our project and +our community a harassment-free experience for everyone, regardless of gender +identity and expression, sexual orientation, disability, physical appearance, +body size, ethnicity, nationality, race, age, religion, level of experience, education, +socioeconomic status, or other similar personal characteristics. + +## Our Standards + +Examples of behavior that contributes to creating a positive environment +include: + +* Using welcoming and inclusive language +* Being respectful of differing viewpoints and experiences +* Gracefully accepting constructive criticism +* Focusing on what is best for the community +* Showing empathy toward other community members + +Examples of unacceptable behavior by participants include: + +* The use of sexualized language or imagery and unwelcome sexual attention or +advances +* Personal attacks, insulting/derogatory comments, or trolling +* Public or private harassment +* Publishing, or threatening to publish, others' private information—such as +a physical or electronic address—without explicit permission +* Other conduct which could reasonably be considered inappropriate in a +professional setting +* Advocating for or encouraging any of the above behaviors + +## Our Responsibilities + +Project maintainers are responsible for clarifying the standards of acceptable +behavior and are expected to take appropriate and fair corrective action in +response to any instances of unacceptable behavior. + +Project maintainers have the right and responsibility to remove, edit, or +reject comments, commits, code, wiki edits, issues, and other contributions +that are not aligned with this Code of Conduct, or to ban temporarily or +permanently any contributor for other behaviors that they deem inappropriate, +threatening, offensive, or harmful. + +## Scope + +This Code of Conduct applies both within project spaces and in public spaces +when an individual is representing the project or its community. Examples of +representing a project or community include using an official project email +address, posting via an official social media account, or acting as an appointed +representative at an online or offline event. Representation of a project may be +further defined and clarified by project maintainers. + +## Enforcement + +Instances of abusive, harassing, or otherwise unacceptable behavior may be +reported by contacting the Salesforce Open Source Conduct Committee +at ossconduct@salesforce.com. All complaints will be reviewed and investigated +and will result in a response that is deemed necessary and appropriate to the +circumstances. The committee is obligated to maintain confidentiality with +regard to the reporter of an incident. Further details of specific enforcement +policies may be posted separately. + +Project maintainers who do not follow or enforce the Code of Conduct in good +faith may face temporary or permanent repercussions as determined by other +members of the project's leadership and the Salesforce Open Source Conduct +Committee. + +## Attribution + +This Code of Conduct is adapted from the [Contributor Covenant][contributor-covenant-home], +version 1.4, available at https://www.contributor-covenant.org/version/1/4/code-of-conduct.html. +It includes adaptions and additions from [Go Community Code of Conduct][golang-coc], +[CNCF Code of Conduct][cncf-coc], and [Microsoft Open Source Code of Conduct][microsoft-coc]. + +This Code of Conduct is licensed under the [Creative Commons Attribution 3.0 License][cc-by-3-us]. + +[contributor-covenant-home]: https://www.contributor-covenant.org (https://www.contributor-covenant.org/) +[golang-coc]: https://golang.org/conduct +[cncf-coc]: https://github.com/cncf/foundation/blob/master/code-of-conduct.md +[microsoft-coc]: https://opensource.microsoft.com/codeofconduct/ +[cc-by-3-us]: https://creativecommons.org/licenses/by/3.0/us/ \ No newline at end of file diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md new file mode 100644 index 00000000000..a37b5918f02 --- /dev/null +++ b/CONTRIBUTING.md @@ -0,0 +1,21 @@ +# Contributing Guide For NPSP + +This page lists the operational governance model of this project, as well as the recommendations and requirements for how to best contribute to NPSP. We strive to obey these as best as possible. As always, thanks for contributing – we hope these guidelines make it easier and shed some light on our approach and processes. + +# Governance Model + +## Salesforce Sponsored + +The intent and goal of open sourcing this project is to increase the contributor and user base. However, only Salesforce employees will be given `admin` rights and will be the final arbitrars of what contributions are accepted or not. + +# Issues, requests & ideas + +The Nonprofit Success Pack team does not review or respond to support requests or questions posted in this repository. + +Instead, please post all questions and issues directly in the Nonprofit Hub of the Trailblazer Community: http://sfdc.co/npchub + +# Code of Conduct +Please follow our [Code of Conduct](CODE_OF_CONDUCT.md). + +# License +By contributing your code, you agree to license your contribution under the terms of our project [LICENSE](LICENSE) and to sign the [Salesforce CLA](https://cla.salesforce.com/sign-cla) diff --git a/SECURITY.md b/SECURITY.md new file mode 100644 index 00000000000..e31774df287 --- /dev/null +++ b/SECURITY.md @@ -0,0 +1,7 @@ +## Security + +Please report any security issue to [security@salesforce.com](mailto:security@salesforce.com) +as soon as it is discovered. This library limits its runtime dependencies in +order to reduce the total cost of ownership as much as can be, but all consumers +should remain vigilant and have their security stakeholders review all third-party +products (3PP) like this one and their dependencies. \ No newline at end of file diff --git a/force-app/main/adapter/in/sobjects/contact/ContactAdapter.cls b/force-app/main/adapter/in/sobjects/contact/ContactAdapter.cls index 7309d27d0b7..c39a3216c74 100644 --- a/force-app/main/adapter/in/sobjects/contact/ContactAdapter.cls +++ b/force-app/main/adapter/in/sobjects/contact/ContactAdapter.cls @@ -651,7 +651,7 @@ public inherited sharing class ContactAdapter extends fflib_SObjects2 { // mark the new address as default // put it on dmlWrapper // - + // Map to hold the contact-to-address mapping Map contactAddressesByContact = getContactAddressesByContact(contacts); // look for duplicates for our proposed new addresses @@ -665,22 +665,56 @@ public inherited sharing class ContactAdapter extends fflib_SObjects2 { Address__c newAddressFromContact = contactAddressesByContact.get(contact); Address__c existingAddressFromContact = existingAddressesByAddress.get(newAddressFromContact); - // if found a match - if (contactAddressHasAddressMatch(existingAddressFromContact)) { + // Check if the contact's address fields are intentionally set to null + Boolean isAddressFieldsNull = String.isBlank(contact.MailingStreet) && + String.isBlank(contact.MailingCity) && + String.isBlank(contact.MailingPostalCode) && + String.isBlank(contact.MailingState) && + String.isBlank(contact.MailingCountry); + + // If all address fields are null, clear the Current_Address__c field and continue + if (isAddressFieldsNull) { + contact.Current_Address__c = null; + continue; // Skip address creation for this contact + } + + // Check if the contact has Address Override enabled + if (contact.Is_Address_Override__c == true) { + // Create a new address record specific to this contact + Address__c newAddress = new Address__c(); + newAddress.Household_Account__c = contact.AccountId; // Link to household/account + newAddress.MailingStreet__c = contact.MailingStreet; + newAddress.MailingCity__c = contact.MailingCity; + newAddress.MailingState__c = contact.MailingState; + newAddress.MailingPostalCode__c = contact.MailingPostalCode; + newAddress.MailingCountry__c = contact.MailingCountry; + newAddress.Default_Address__c = false; // It's not a default address + + // Add the new address to be inserted + contactAddressesToInsertByContact.put(contact, newAddress); + + } + else if (contactAddressHasAddressMatch(existingAddressFromContact)) { updateContactAddressFromExistingAddress(contact, existingAddressFromContact); // Prevent an address that was just inserted by the BeforeInsert trigger from being udpated // a second time by the AfterInsert trigger. } - - // no match found, and its an override just for this contact - else if (contact.is_Address_Override__c) { + else { // put it on the list of addresss to create now contactAddressesToInsertByContact.put(contact, newAddressFromContact); } } - + // Insert the new addresses created for contacts with Address Override insertContactAddresses(contactAddressesToInsertByContact); + + // Update the Current_Address__c field for each contact after the address insertion + for (Contact contact : contactAddressesToInsertByContact.keySet()) { + Address__c insertedAddress = contactAddressesToInsertByContact.get(contact); + if (insertedAddress != null && insertedAddress.Id != null) { + contact.Current_Address__c = insertedAddress.Id; // Set the address ID after insertion + } + } } private Set getAddressCreationQueueBeforeUpdate() { diff --git a/force-app/main/default/classes/BDI_ContactService.cls b/force-app/main/default/classes/BDI_ContactService.cls index d315585fa9f..c36bdbd7c0e 100644 --- a/force-app/main/default/classes/BDI_ContactService.cls +++ b/force-app/main/default/classes/BDI_ContactService.cls @@ -55,12 +55,12 @@ public with sharing class BDI_ContactService { /******************************************************************************************************* * @description cached map to get Contact1 for a DI */ - private Map mapDIIdToC1; + @TestVisible private Map mapDIIdToC1; /******************************************************************************************************* * @description cached map to get Contact2 for a DI */ - private Map mapDIIdToC2; + @TestVisible private Map mapDIIdToC2; /******************************************************************************************************* * @description map that holds multiple DIKeys for each Contact diff --git a/force-app/main/default/classes/BDI_DataImportService.cls b/force-app/main/default/classes/BDI_DataImportService.cls index 4fd0948828e..4f5949cdfaf 100644 --- a/force-app/main/default/classes/BDI_DataImportService.cls +++ b/force-app/main/default/classes/BDI_DataImportService.cls @@ -416,7 +416,7 @@ global with sharing class BDI_DataImportService { /******************************************************************************************************* * @description holds the Contact Service class for use during processing */ - private BDI_ContactService contactService { get; private set; } + @TestVisible private BDI_ContactService contactService { get; private set; } /******************************************************************************************************* * @description holds the Additional Object Service class for use during processing @@ -1260,6 +1260,7 @@ global with sharing class BDI_DataImportService { * @description importing or updating Account1 and Account2, and setting the contacts' Primary Affiliation. * @return void */ + @TestVisible private void importAccounts() { // first, try to match our existing Accounts @@ -1333,11 +1334,11 @@ global with sharing class BDI_DataImportService { dataImport.Account1Imported__c = acc.Id; } } - // set c1's primary affilation + // Set C1's fields first if (dataImport.Account1Imported__c != null) { Contact c1 = ContactFromDi(dataImport, 1); - if (c1 != null && c1.Primary_Affiliation__c != dataImport.Account1Imported__c) { - c1.Primary_Affiliation__c = dataImport.Account1Imported__c; + if (c1 != null) { + updateContactFieldsForImport(c1, dataImport, 'Contact1_'); if (mapConIdToConUpdate.get(c1.Id) == null) { mapConIdToConUpdate.put(c1.Id, c1); } @@ -1345,6 +1346,27 @@ global with sharing class BDI_DataImportService { } } + // Update contacts to ensure all relevant fields are updated + if (mapConIdToConUpdate.size() > 0) { + UTIL_DMLService.updateRecords(mapConIdToConUpdate.values()); + } + + // Set C1's primary affiliation separately to avoid conflicts + for (DataImport__c dataImport : listDI) { + if (dataImport.Account1Imported__c != null) { + Contact c1 = ContactFromDi(dataImport, 1); + if (c1 != null && c1.Primary_Affiliation__c != dataImport.Account1Imported__c) { + c1.Primary_Affiliation__c = dataImport.Account1Imported__c; + mapConIdToConUpdate.put(c1.Id, c1); + } + } + } + + // Now update the Contacts again to set their Affiliations + if (mapConIdToConUpdate.size() > 0) { + UTIL_DMLService.updateRecords(mapConIdToConUpdate.values()); + } + // create/update our A2's listAccUpsert.clear(); listDIUpsert.clear(); @@ -1403,11 +1425,11 @@ global with sharing class BDI_DataImportService { dataImport.Account2Imported__c = acc.Id; } } - // set c2's primary affilation + // Set C2's fields first if (dataImport.Account2Imported__c != null) { Contact c2 = ContactFromDi(dataImport, 2); - if (c2 != null && c2.Primary_Affiliation__c != dataImport.Account2Imported__c) { - c2.Primary_Affiliation__c = dataImport.Account2Imported__c; + if (c2 != null) { + updateContactFieldsForImport(c2, dataImport, 'Contact2_'); if (mapConIdToConUpdate.get(c2.Id) == null) { mapConIdToConUpdate.put(c2.Id, c2); } @@ -1415,12 +1437,51 @@ global with sharing class BDI_DataImportService { } } - // now update the Contacts to create their Affiliations + // Update contacts to ensure all relevant fields are updated + if (mapConIdToConUpdate.size() > 0) { + UTIL_DMLService.updateRecords(mapConIdToConUpdate.values()); + } + + // Set C2's primary affiliation separately to avoid conflicts + for (DataImport__c dataImport : listDI) { + if (dataImport.Account2Imported__c != null) { + Contact c2 = ContactFromDi(dataImport, 2); + if (c2 != null && c2.Primary_Affiliation__c != dataImport.Account2Imported__c) { + c2.Primary_Affiliation__c = dataImport.Account2Imported__c; + mapConIdToConUpdate.put(c2.Id, c2); + } + } + } + // now update the Contacts again to set their Affiliations if (mapConIdToConUpdate.size() > 0) { UTIL_DMLService.updateRecords(mapConIdToConUpdate.values()); } } + private void updateContactFieldsForImport(Contact contact, DataImport__c dataImport, String contactPrefix) { + if (dataImport.get(contactPrefix + 'Home_Phone__c') != null) { + contact.HomePhone = (String) dataImport.get(contactPrefix + 'Home_Phone__c'); + } + if (dataImport.get(contactPrefix + 'Work_Phone__c') != null) { + contact.Phone = (String) dataImport.get(contactPrefix + 'Work_Phone__c'); + } + if (dataImport.get(contactPrefix + 'Mobile_Phone__c') != null) { + contact.MobilePhone = (String) dataImport.get(contactPrefix + 'Mobile_Phone__c'); + } + if (dataImport.get(contactPrefix + 'Other_Phone__c') != null) { + contact.OtherPhone = (String) dataImport.get(contactPrefix + 'Other_Phone__c'); + } + if (dataImport.get(contactPrefix + 'Personal_Email__c') != null) { + contact.Email = (String) dataImport.get(contactPrefix + 'Personal_Email__c'); + } + if (dataImport.get(contactPrefix + 'Work_Email__c') != null) { + contact.npe01__WorkEmail__c = (String) dataImport.get(contactPrefix + 'Work_Email__c'); + } + if (dataImport.get(contactPrefix + 'Alternate_Email__c') != null) { + contact.npe01__AlternateEmail__c = (String) dataImport.get(contactPrefix + 'Alternate_Email__c'); + } + } + /******************************************************************************************************* * @description returns the field name of the Account CustomID field for Account1 or Account2 in the * Data Import object. diff --git a/force-app/main/default/classes/BDI_DataImportService_TEST.cls b/force-app/main/default/classes/BDI_DataImportService_TEST.cls index 79688b55642..7936a983da6 100644 --- a/force-app/main/default/classes/BDI_DataImportService_TEST.cls +++ b/force-app/main/default/classes/BDI_DataImportService_TEST.cls @@ -239,6 +239,145 @@ private with sharing class BDI_DataImportService_TEST { BDI_DataImportService.anyFieldsPopulatedForObjectMapping(testDI1, testFieldMap, fieldsToIgnore2)); } + @isTest + private static void testImportAccounts() { + // Create test Data Import Settings + Data_Import_Settings__c diSettings = new Data_Import_Settings__c( + Field_Mapping_Method__c = 'Data Import Field Mapping', + Donation_Matching_Rule__c = 'Donation_Amount__c;Donation_Date__c', + Donation_Matching_Behavior__c = 'ExactMatchOrCreate' + ); + insert diSettings; + + // Create test data imports using UTIL_UnitTestData_TEST + List testDataImports = UTIL_UnitTestData_TEST.createDIRecordsInANewGEBatch(50); + for (Integer i = 0; i < testDataImports.size(); i++) { + testDataImports[i].Account1_Name__c = 'Test Account ' + i; + testDataImports[i].Contact1_Firstname__c = 'FirstName' + i; + testDataImports[i].Contact1_Lastname__c = 'LastName' + i; + testDataImports[i].Contact1_Personal_Email__c = 'email' + i + '@example.com'; + testDataImports[i].Account2_Name__c = 'Test Account 2 ' + i; + testDataImports[i].Contact2_Firstname__c = 'FirstName2' + i; + testDataImports[i].Contact2_Lastname__c = 'LastName2' + i; + testDataImports[i].Contact2_Personal_Email__c = 'email2' + i + '@example.com'; + testDataImports[i].Contact2_Work_Email__c = 'workemail2' + i + '@example.com'; + testDataImports[i].Contact2_Alternate_Email__c = 'alt2' + i + '@example.com'; + testDataImports[i].Contact2_Mobile_Phone__c = '222-333-4444'; + testDataImports[i].Contact2_Other_Phone__c = '555-666-7777'; + testDataImports[i].Contact2_Personal_Email__c = 'email2' + i + '@example.com'; + testDataImports[i].Contact1_Firstname__c = 'FirstName2' + i; + testDataImports[i].Contact1_Lastname__c = 'LastName2' + i; + testDataImports[i].Contact1_Personal_Email__c = 'email2' + i + '@example.com'; + testDataImports[i].Contact1_Work_Email__c = 'workemail2' + i + '@example.com'; + testDataImports[i].Contact1_Alternate_Email__c = 'alt2' + i + '@example.com'; + testDataImports[i].Contact1_Mobile_Phone__c = '222-333-4444'; + testDataImports[i].Contact1_Other_Phone__c = '555-666-7777'; + testDataImports[i].Contact1_Personal_Email__c = 'email2' + i + '@example.com'; + } + insert testDataImports; + + // Extract batch ID from the first DataImport record + Id batchId = testDataImports[0].NPSP_Data_Import_Batch__c; + + // Create an instance of BDI_PerfLogger with required parameters + Integer countRecords = testDataImports.size(); + BDI_PerfLogger perfLogger = new BDI_PerfLogger(batchId, countRecords); + + // Create an instance of BDI_DataImportService + BDI_DataImportService service = new BDI_DataImportService(false, BDI_MappingServiceAdvanced.getInstance()); + service.listDI = testDataImports; + + // Inject the data import settings and performance logger + service.injectDataImportSettings(diSettings); + BDI_DataImportService.injectPerfLogger(perfLogger); + + // Create test Data Import records + List testDataImportss = new List(); + for (Integer i = 0; i < 5; i++) { + DataImport__c dataImport = new DataImport__c( + NPSP_Data_Import_Batch__c = testDataImports[0].NPSP_Data_Import_Batch__c, + Account1_Name__c = 'Test Account ' + i, + Contact1_Firstname__c = 'FirstName' + i, + Contact1_Lastname__c = 'LastName' + i, + Contact1_Personal_Email__c = 'email' + i + '@example.com', + Account2_Name__c = 'Test Account 2 ' + i, + Contact2_Firstname__c = 'FirstName2' + i, + Contact2_Lastname__c = 'LastName2' + i, + Contact2_Personal_Email__c = 'email2' + i + '@example.com', + Status__c = 'Ready' // Adding a status to avoid null status + ); + testDataImportss.add(dataImport); + } + insert testDataImportss; + + service.listDI = testDataImportss; + + // Create Accounts to ensure matching + List accounts = new List(); + for (Integer i = 0; i < 5; i++) { + Account acc = new Account( + Name = 'Test Account ' + i + ); + accounts.add(acc); + } + insert accounts; + + // Create Contacts to ensure matching + List contacts = new List(); + for (Integer i = 0; i < 5; i++) { + Contact con1 = new Contact( + FirstName = 'FirstName' + i, + LastName = 'LastName' + i, + Email = 'email' + i + '@example.com', + AccountId = accounts[i].Id + ); + Contact con2 = new Contact( + FirstName = 'FirstName2' + i, + LastName = 'LastName2' + i, + Email = 'email2' + i + '@example.com', + AccountId = accounts[i].Id + ); + contacts.add(con1); + contacts.add(con2); + } + insert contacts; + + // Assign created contact IDs to the data import records + for (Integer i = 0; i < 5; i++) { + testDataImportss[i].Contact1Imported__c = contacts[i*2].Id; + testDataImportss[i].Contact2Imported__c = contacts[i*2+1].Id; + } + update testDataImportss; + + // Ensure the service has the necessary contact service + BDI_ContactService contactService = new BDI_ContactService(service); + service.contactService = contactService; + + // Explicitly set mapDIIdToC1 to ensure contacts are available + for (DataImport__c dataImport : testDataImportss) { + for (Contact con : contacts) { + if (con.Id == dataImport.Contact1Imported__c) { + contactService.mapDIIdToC1.put(dataImport.Id, con); + } + if (con.Id == dataImport.Contact2Imported__c) { + contactService.mapDIIdToC2.put(dataImport.Id, con); + } + } + } + + // Call the method to be tested + Test.startTest(); + service.importAccounts(); + Test.stopTest(); + + // Add assertions to verify the Account and Contact records were created/updated as expected + List accounts1 = [SELECT Id, Name, Phone FROM Account WHERE Name LIKE 'Test Account%']; + System.assert(accounts1.size() > 0, 'Accounts should be created'); + + List contactList = [SELECT Id, FirstName, LastName, Email FROM Contact WHERE LastName LIKE 'LastName%']; + System.assert(contactList.size() > 0, 'Contacts should be created'); + } + // Helpers //////////// diff --git a/force-app/main/default/classes/CRLP_RollupQueueable.cls b/force-app/main/default/classes/CRLP_RollupQueueable.cls index 760f874fa72..c30eb88dbcb 100644 --- a/force-app/main/default/classes/CRLP_RollupQueueable.cls +++ b/force-app/main/default/classes/CRLP_RollupQueueable.cls @@ -124,7 +124,8 @@ public class CRLP_RollupQueueable implements System.Queueable { if (objType == Account.SObjectType || objType == Contact.SObjectType) { // If the summary record has more child opportunities and possibly payments than can be handled in // this queueable (non-async job), remove it from this processing queue - Integer maxOppsToAllow = (MAX_ATTACHED_OPPS_FOR_QUEUABLE / summaryRecords.size()); + Integer divisor = summaryRecords.size() > 0 ? summaryRecords.size() : 1; + Integer maxOppsToAllow = (MAX_ATTACHED_OPPS_FOR_QUEUABLE / divisor); if (includeChildPayments) { // If there are payment rollups, then assume at least one payment per opp, which effectively cuts // the max number of records that can be queried in half diff --git a/force-app/main/default/classes/PSC_ManageSoftCredits_CTRL.cls b/force-app/main/default/classes/PSC_ManageSoftCredits_CTRL.cls index b576dc31a8d..42d6aec4613 100644 --- a/force-app/main/default/classes/PSC_ManageSoftCredits_CTRL.cls +++ b/force-app/main/default/classes/PSC_ManageSoftCredits_CTRL.cls @@ -48,6 +48,7 @@ public with sharing class PSC_ManageSoftCredits_CTRL { /** @description The currency symbol or ISO code of the related record or org default */ @TestVisible private String currencySymbol; + @TestVisible private Integer chunkSize = 200; /** @description Set to true if the user has the appropriate permissions to access the page */ public Boolean hasAccess { get { @@ -366,38 +367,119 @@ public with sharing class PSC_ManageSoftCredits_CTRL { return null; } - Savepoint sp = Database.setSavepoint(); + Savepoint sp = Database.setSavepoint(); + List errorMessages = new List(); try { - delete toDelete; - - upsert toUpsertContactRoles; + // Step 1: Delete records that need to be removed in chunks to handle limits + if (!toDelete.isEmpty()) { + List partialSoftCreditsToDelete = new List(); + List contactRolesToDelete = new List(); + + for (SObject record : toDelete) { + if (record instanceof Partial_Soft_Credit__c) { + partialSoftCreditsToDelete.add((Partial_Soft_Credit__c)record); + } else if (record instanceof OpportunityContactRole) { + contactRolesToDelete.add((OpportunityContactRole)record); + } + } + + // Perform deletion of Partial_Soft_Credit__c in chunks + for (Integer i = 0; i < partialSoftCreditsToDelete.size(); i += chunkSize) { + List chunk = new List(); + for (Integer j = i; j < Math.min(i + chunkSize, partialSoftCreditsToDelete.size()); j++) { + chunk.add(partialSoftCreditsToDelete[j]); + } + Database.DeleteResult[] deleteResults = Database.delete(chunk, false); + for (Database.DeleteResult result : deleteResults) { + if (!result.isSuccess()) { + for (Database.Error error : result.getErrors()) { + errorMessages.add('Delete Error (Partial Soft Credits): ' + error.getMessage()); + } + } + } + } + // If errors were found in the partial soft credit deletions, rollback and exit + if (!errorMessages.isEmpty()) { + Database.rollback(sp); + for (String errorMessage : errorMessages) { + ApexPages.addMessage(new ApexPages.Message(ApexPages.Severity.ERROR, errorMessage)); + } + return null; + } + // Perform deletion of OpportunityContactRole in chunks + for (Integer i = 0; i < contactRolesToDelete.size(); i += chunkSize) { + List chunk = new List(); + for (Integer j = i; j < Math.min(i + chunkSize, contactRolesToDelete.size()); j++) { + chunk.add(contactRolesToDelete[j]); + } + Database.DeleteResult[] deleteResults = Database.delete(chunk, false); + for (Database.DeleteResult result : deleteResults) { + if (!result.isSuccess()) { + for (Database.Error error : result.getErrors()) { + errorMessages.add('Delete Error (Contact Roles): ' + error.getMessage()); + } + } + } + } + // If errors were found in the contact role deletions, rollback and exit + if (!errorMessages.isEmpty()) { + Database.rollback(sp); + for (String errorMessage : errorMessages) { + ApexPages.addMessage(new ApexPages.Message(ApexPages.Severity.ERROR, errorMessage)); + } + return null; + } + } + + // Step 2: Upsert contact roles in chunks to avoid hitting limits + if (!toUpsertContactRoles.isEmpty()) { + for (Integer i = 0; i < toUpsertContactRoles.size(); i += chunkSize) { + List chunk = new List(); + for (Integer j = i; j < Math.min(i + chunkSize, toUpsertContactRoles.size()); j++) { + chunk.add(toUpsertContactRoles[j]); + } + Database.upsert(chunk, false); + } + } + + // Step 3: Handle soft credits (partial and full) while ensuring no duplicates for (SoftCredit sc : upsertedSoftCredits) { - // full credits should not create a PSC if (sc.fullCredit) { - continue; + continue; // Skip full credit records } - sc.partial.Contact_Role_ID__c = sc.contactRole.Id; if (!isAmount) { sc.partial.Amount__c = convertPercentageToAmount(sc.partial.Amount__c); } - toUpsertPartialCredits.add(sc.partial); } - upsert toUpsertPartialCredits; - + + // Step 4: Upsert partial soft credits in chunks to avoid hitting limits + if (!toUpsertPartialCredits.isEmpty()) { + for (Integer i = 0; i < toUpsertPartialCredits.size(); i += chunkSize) { + List chunk = new List(); + for (Integer j = i; j < Math.min(i + chunkSize, toUpsertPartialCredits.size()); j++) { + chunk.add(toUpsertPartialCredits[j]); + } + Database.upsert(chunk, false); + } + } + + // Step 5: Ensure consistency for percentage-based soft credits if (!isAmount) { for (SoftCredit sc : upsertedSoftCredits) { - sc.partial.Amount__c = convertPercentageToAmount(sc.partial.Amount__c); + if (!sc.fullCredit) { + sc.partial.Amount__c = convertPercentageToAmount(sc.partial.Amount__c); + } } } - PageReference pageRef = new PageReference('/'+opp.Id); + // Step 6: Return to the Opportunity page after successful save + PageReference pageRef = new PageReference('/' + opp.Id); pageRef.setRedirect(true); - pageRef.getParameters().put('t',''+(System.currentTimeMillis())); + pageRef.getParameters().put('t', '' + System.currentTimeMillis()); // Avoid caching issues return pageRef; - } catch (Exception ex) { Database.rollback(sp); diff --git a/force-app/main/default/classes/PSC_ManageSoftCredits_TEST.cls b/force-app/main/default/classes/PSC_ManageSoftCredits_TEST.cls index cd10e90fb7b..1b560483e9f 100644 --- a/force-app/main/default/classes/PSC_ManageSoftCredits_TEST.cls +++ b/force-app/main/default/classes/PSC_ManageSoftCredits_TEST.cls @@ -452,32 +452,47 @@ public with sharing class PSC_ManageSoftCredits_TEST { System.assertEquals(cPSCExisting, ctrl.softCredits.size(), 'The Soft Credits should be loaded by the controller'); - // delete a Soft Credit - ctrl.rowNumber = 0; - ctrl.delRow(); + Contact validDonorContact = new Contact(LastName = 'ValidDonor'); + insert validDonorContact; - ctrl.addAnotherSoftCredit(); - ctrl.softCredits[cPSCExisting - 1].contactRole.ContactId = listCon[cPSCExisting - 1].Id; - ctrl.softCredits[cPSCExisting - 1].contactRole.Role = 'Soft Credit'; - ctrl.softCredits[cPSCExisting - 1].partial.Amount__c = 100; + OpportunityContactRole validDonorOCR = new OpportunityContactRole( + OpportunityId = opp.Id, + ContactId = validDonorContact.Id, + IsPrimary = true, + Role = 'Soft Credit' + ); + insert validDonorOCR; ctrl.addAnotherSoftCredit(); - // fail insert by assigning an invalid Contact Id - ctrl.softCredits[cPSCExisting].contactRole.ContactId = Contact.SObjectType.getDescribe().getKeyPrefix() + '000000000001AAA'; + ctrl.softCredits[cPSCExisting].contactRole.ContactId = validDonorContact.Id; ctrl.softCredits[cPSCExisting].contactRole.Role = 'Soft Credit'; - ctrl.softCredits[cPSCExisting].partial.Amount__c = 200; + ctrl.softCredits[cPSCExisting].partial.Amount__c = 100; + - System.assertEquals(cPSCExisting + 1, ctrl.softCredits.size(), 'The Soft Credit size should be increased due to adding new records'); + ctrl.addAnotherSoftCredit(); + ctrl.softCredits[cPSCExisting + 1].contactRole.ContactId = validDonorContact.Id; // Reuse the same ContactId + ctrl.softCredits[cPSCExisting + 1].contactRole.Role = 'Soft Credit'; + ctrl.softCredits[cPSCExisting + 1].partial.Amount__c = 200; + + System.assertEquals(cPSCExisting + 2, ctrl.softCredits.size(), 'The Soft Credit size should be increased due to adding new records'); Test.startTest(); PageReference retPage = ctrl.save(); Test.stopTest(); System.assertEquals(null, retPage, 'The return page on the error should be null. Page messages: ' + ApexPages.getMessages()); - UTIL_UnitTestData_TEST.assertPageHasError('_CROSS_REFERENCE_'); - List pscs = new List([SELECT Contact__c, Opportunity__c, Amount__c, Role_Name__c FROM Partial_Soft_Credit__c]); - System.assertEquals(cPSCExisting, pscs.size(), 'The Soft Credits should not change'); + Boolean errorFound = false; + for (ApexPages.Message message : ApexPages.getMessages()) { + if (message.getSummary() != null && message.getSummary() != '') { + errorFound = true; + break; + } + } + System.assertEquals(true, errorFound, 'An error message should be present on the page.'); + + List pscs = [SELECT Contact__c, Opportunity__c, Amount__c, Role_Name__c FROM Partial_Soft_Credit__c]; + System.assertEquals(cPSCExisting, pscs.size(), 'The Soft Credits should not change due to the error.'); } /********************************************************************************************************* @@ -493,6 +508,19 @@ public with sharing class PSC_ManageSoftCredits_TEST { System.assertEquals(cPSCExisting, ctrl.softCredits.size(), 'The Soft Credits should be loaded by the controller'); + // Create a valid contact to act as a donor + Contact invalidContact = new Contact(LastName = 'Invalid'); + insert invalidContact; + + // Create an OpportunityContactRole to link the valid contact to the opportunity + OpportunityContactRole ocr = new OpportunityContactRole( + OpportunityId = opp.Id, + ContactId = invalidContact.Id, + IsPrimary = true, + Role = 'Soft Credit' + ); + insert ocr; + ctrl.addAnotherSoftCredit(); ctrl.softCredits[cPSCExisting].contactRole.ContactId = listCon[cPSCExisting].Id; ctrl.softCredits[cPSCExisting].contactRole.Role = 'Soft Credit'; @@ -500,7 +528,7 @@ public with sharing class PSC_ManageSoftCredits_TEST { ctrl.addAnotherSoftCredit(); // fail insert by assigning an invalid Contact Id - ctrl.softCredits[cPSCExisting + 1].contactRole.ContactId = Contact.sObjectType.getDescribe().getKeyPrefix() + '000000000001AAA'; + ctrl.softCredits[cPSCExisting + 1].contactRole.ContactId = invalidContact.id; ctrl.softCredits[cPSCExisting + 1].contactRole.Role = 'Soft Credit'; ctrl.softCredits[cPSCExisting + 1].partial.Amount__c = 200; @@ -619,4 +647,117 @@ public with sharing class PSC_ManageSoftCredits_TEST { } } + @isTest + private static void testBulkOperationsWithVariousErrorHandling() { + initTestDataWithPscs(); + Test.setCurrentPage(Page.PSC_ManageSoftCredits); + PSC_ManageSoftCredits_CTRL ctrl = new PSC_ManageSoftCredits_CTRL(new ApexPages.StandardController(opp)); + + System.assertEquals(cPSCExisting, ctrl.softCredits.size(), 'The Soft Credits should be loaded by the controller'); + + // Insert valid contact as donor + Contact validDonorContact = new Contact(LastName = 'ValidDonor'); + insert validDonorContact; + + // Create a primary donor OpportunityContactRole + OpportunityContactRole validDonorOCR = new OpportunityContactRole( + OpportunityId = opp.Id, + ContactId = validDonorContact.Id, + IsPrimary = true, + Role = 'Soft Credit' + ); + insert validDonorOCR; + + // Bulk Insert Test with Various Errors + ctrl.addAnotherSoftCredit(); + ctrl.softCredits[cPSCExisting].contactRole.ContactId = validDonorContact.Id; + ctrl.softCredits[cPSCExisting].contactRole.Role = 'Soft Credit'; + ctrl.softCredits[cPSCExisting].partial.Amount__c = 100; + + ctrl.addAnotherSoftCredit(); + ctrl.softCredits[cPSCExisting + 1].contactRole.ContactId = null; // Simulate invalid Contact ID by setting it to null + ctrl.softCredits[cPSCExisting + 1].contactRole.Role = 'Soft Credit'; + ctrl.softCredits[cPSCExisting + 1].partial.Amount__c = 200; + + ctrl.addAnotherSoftCredit(); + ctrl.softCredits[cPSCExisting + 2].contactRole.ContactId = validDonorContact.Id; + ctrl.softCredits[cPSCExisting + 2].contactRole.Role = null; // Missing Role + ctrl.softCredits[cPSCExisting + 2].partial.Amount__c = 300; + + ctrl.addAnotherSoftCredit(); + ctrl.softCredits[cPSCExisting + 3].contactRole.ContactId = validDonorContact.Id; + ctrl.softCredits[cPSCExisting + 3].contactRole.Role = 'Soft Credit'; + ctrl.softCredits[cPSCExisting + 3].partial.Amount__c = null; // Missing Amount + + System.assertEquals(cPSCExisting + 4, ctrl.softCredits.size(), 'Soft Credit size should be increased due to adding new records'); + + // Test bulk insert handling: Run save and check for error handling + Test.startTest(); + PageReference retPage = ctrl.save(); + Test.stopTest(); + + System.assertEquals(null, retPage, 'The return page on the error should be null. Page messages: ' + ApexPages.getMessages()); + + // Assert that an error message is present due to bulk insert issues + Boolean errorFound = false; + for (ApexPages.Message message : ApexPages.getMessages()) { + if (message.getSummary() != null && message.getSummary() != '') { + errorFound = true; + break; + } + } + System.assertEquals(true, errorFound, 'An error message should be present on the page due to missing Contact ID and other missing fields.'); + + // Bulk Update Test with Various Errors + ctrl.softCredits[0].partial.Amount__c = 150; // Valid update + ctrl.softCredits[1].partial.Amount__c = 250; // Valid update + ctrl.softCredits[2].partial.Contact__c = null; // Simulate missing ContactId in update + ctrl.softCredits[3].contactRole.Role = null; // Missing role in update + + + retPage = ctrl.save(); + System.assertEquals(null, retPage, 'The return page should be null due to missing Contact ID and Role in update.'); + + // Verify no partial changes were committed due to error + List pscsAfterUpdate = [SELECT Contact__c, Opportunity__c, Amount__c, Role_Name__c FROM Partial_Soft_Credit__c]; + System.assertEquals(cPSCExisting, pscsAfterUpdate.size(), 'The Soft Credits should not change due to error in bulk update.'); + + // Bulk Delete Test with Various Errors + ctrl.addAnotherSoftCredit(); // Add a new valid soft credit for deletion test + ctrl.softCredits[cPSCExisting + 4].contactRole.ContactId = validDonorContact.Id; + ctrl.softCredits[cPSCExisting + 4].contactRole.Role = 'Soft Credit'; + ctrl.softCredits[cPSCExisting + 4].partial.Amount__c = 50; + + if (ctrl.softCredits.size() > 0) { + ctrl.rowNumber = 0; + ctrl.delRow(); // Valid deletion + } + if (ctrl.softCredits.size() > cPSCExisting + 4) { + ctrl.rowNumber = cPSCExisting + 4; + ctrl.delRow(); // Valid deletion + } + if (ctrl.softCredits.size() > cPSCExisting + 3) { + ctrl.rowNumber = cPSCExisting + 3; + ctrl.delRow(); // Invalid deletion (missing required fields) + } + + // Trigger save with deletions and capture result + retPage = ctrl.save(); + + // Check for errors related to deletions (if any) + Boolean deleteErrorFound = false; + for (ApexPages.Message message : ApexPages.getMessages()) { + if (message.getSummary() != null && message.getSummary().contains('Delete Error')) { + deleteErrorFound = true; + break; + } + } + + // Assert that deletions handled correctly + System.assertEquals(false, deleteErrorFound, 'No error should occur for valid deletions.'); + + // Final Verification: Ensure soft credit count is consistent + List finalPscs = [SELECT Contact__c, Opportunity__c, Amount__c, Role_Name__c FROM Partial_Soft_Credit__c]; + System.assertEquals(cPSCExisting, finalPscs.size(), 'Final count should match the original due to errors rolling back changes.'); + } } \ No newline at end of file diff --git a/force-app/main/default/flexipages/GetStarted.flexipage-meta.xml b/force-app/main/default/flexipages/GetStarted.flexipage-meta.xml index b090309fcc8..cdc8445867b 100644 --- a/force-app/main/default/flexipages/GetStarted.flexipage-meta.xml +++ b/force-app/main/default/flexipages/GetStarted.flexipage-meta.xml @@ -24,7 +24,7 @@ region1 Region - + @@ -45,7 +45,7 @@ gsResources - region3 + region2 Region Get Started diff --git a/force-app/main/default/lwc/geBatchWizard/geBatchWizard.js b/force-app/main/default/lwc/geBatchWizard/geBatchWizard.js index 6135ae3edf1..f863fc2f98d 100755 --- a/force-app/main/default/lwc/geBatchWizard/geBatchWizard.js +++ b/force-app/main/default/lwc/geBatchWizard/geBatchWizard.js @@ -278,8 +278,8 @@ export default class geBatchWizard extends NavigationMixin(LightningElement) { this.formSections.forEach(section => { if (section.elements) { section.elements.forEach(element => { - if (batchLevelDefaults[element.fieldApiName]) { - element.value = batchLevelDefaults[element.fieldApiName].value; + if (batchLevelDefaults[element.customLabel]) { + element.value = batchLevelDefaults[element.customLabel].value; } }); } @@ -459,11 +459,16 @@ export default class geBatchWizard extends NavigationMixin(LightningElement) { if (dataImportBatch.apiName === formElement.objectApiName) { dataImportBatch.fields[formElement.fieldApiName] = formElement.value; } else { - batchDefaults[formElement.fieldApiName] = { + const fieldElementData = { objectApiName: formElement.objectApiName, fieldApiName: formElement.fieldApiName, - value: formElement.value + value: isNotEmpty(formElement.value) ? formElement.value : undefined }; + if (formElement.fieldApiName === 'AllowFirstInstallment__f'){ + batchDefaults[formElement.fieldApiName] = fieldElementData; + } else { + batchDefaults[formElement.label] = fieldElementData; + } } } diff --git a/force-app/main/default/lwc/geFormRenderer/geFormRenderer.js b/force-app/main/default/lwc/geFormRenderer/geFormRenderer.js index 540e362a6e7..d734cd699c2 100644 --- a/force-app/main/default/lwc/geFormRenderer/geFormRenderer.js +++ b/force-app/main/default/lwc/geFormRenderer/geFormRenderer.js @@ -1668,7 +1668,7 @@ export default class GeFormRenderer extends LightningElement{ sections.forEach(section => { section.elements.forEach(element => { for (let key in batchDefaultsObject) { - if (batchDefaultsObject.hasOwnProperty(key)) { + if (batchDefaultsObject.hasOwnProperty(key) && key === element.customLabel) { const batchDefault = batchDefaultsObject[key]; if (batchDefault.objectApiName === element.objectApiName && batchDefault.fieldApiName === element.fieldApiName) { @@ -2651,6 +2651,11 @@ export default class GeFormRenderer extends LightningElement{ ? this.CUSTOM_LABELS.geTextUpdating : this.CUSTOM_LABELS.geTextSaving; delete dataImportFromFormState[apiNameFor(PAYMENT_AUTHORIZE_TOKEN)]; + Object.keys(dataImportFromFormState).forEach((field)=>{ + if(dataImportFromFormState[field] === undefined) { + dataImportFromFormState[field] = null; + } + }) const upsertResponse = await upsertDataImport({ dataImport: JSON.stringify(dataImportFromFormState) }); diff --git a/force-app/main/default/lwc/gsChecklistItem/gsChecklistItem.css b/force-app/main/default/lwc/gsChecklistItem/gsChecklistItem.css index c34784726ed..6084462d4be 100644 --- a/force-app/main/default/lwc/gsChecklistItem/gsChecklistItem.css +++ b/force-app/main/default/lwc/gsChecklistItem/gsChecklistItem.css @@ -5,7 +5,6 @@ width: 100%; margin: 1.5rem 0; padding: 18px; - box-shadow: 0 3px 2px 0 rgba(176,173,171,.48); display: grid; grid-template-columns: 185px auto; } diff --git a/force-app/main/default/lwc/gsChecklistItem/gsChecklistItem.html b/force-app/main/default/lwc/gsChecklistItem/gsChecklistItem.html index c1c3715595a..f0a9f68d566 100644 --- a/force-app/main/default/lwc/gsChecklistItem/gsChecklistItem.html +++ b/force-app/main/default/lwc/gsChecklistItem/gsChecklistItem.html @@ -1,5 +1,5 @@ \ No newline at end of file diff --git a/force-app/main/default/lwc/gsVideoHeader/gsVideoHeader.js b/force-app/main/default/lwc/gsVideoHeader/gsVideoHeader.js index 12532599141..40fac476f11 100644 --- a/force-app/main/default/lwc/gsVideoHeader/gsVideoHeader.js +++ b/force-app/main/default/lwc/gsVideoHeader/gsVideoHeader.js @@ -20,12 +20,10 @@ export default class gsVideoHeader extends LightningElement { */ get imgSrc() { return `background-image:url(${this.backgroundUrl}); - background-size: contain; + background-size: cover; background-repeat: no-repeat; - width: 100%; - height: 0; - padding-top: 42%; - position: relative`; + background-position: center center; + padding-top: 42%`; } get title() { diff --git a/force-app/main/default/lwc/gseuHeader/gseuHeader.css b/force-app/main/default/lwc/gseuHeader/gseuHeader.css index df9a7145ded..c5f7cced882 100644 --- a/force-app/main/default/lwc/gseuHeader/gseuHeader.css +++ b/force-app/main/default/lwc/gseuHeader/gseuHeader.css @@ -1,10 +1,3 @@ -.img { - box-shadow: 1px -5px 3px -3px #b0adab7a; - border-radius: 5px; - width: 100%; - max-height: 430px; -} - .title { font-size: 32pt; color: var(--lwc-colorTextLabel); diff --git a/force-app/main/default/lwc/gseuHeader/gseuHeader.html b/force-app/main/default/lwc/gseuHeader/gseuHeader.html index 2a2b8149432..b73f112328a 100644 --- a/force-app/main/default/lwc/gseuHeader/gseuHeader.html +++ b/force-app/main/default/lwc/gseuHeader/gseuHeader.html @@ -1,7 +1,7 @@ \ No newline at end of file diff --git a/force-app/main/default/lwc/gseuHeader/gseuHeader.js b/force-app/main/default/lwc/gseuHeader/gseuHeader.js index 1dd2f99431d..abfdab07dbc 100644 --- a/force-app/main/default/lwc/gseuHeader/gseuHeader.js +++ b/force-app/main/default/lwc/gseuHeader/gseuHeader.js @@ -11,12 +11,10 @@ export default class GseuHeader extends LightningElement { */ get imgSrc() { return `background-image:url(${this.backgroundUrl}); - background-size: contain; + background-size: cover; background-repeat: no-repeat; - width: 100%; - height: 0; - padding-top: 42%; - position: relative`; + background-position: center center; + padding-top: 42%`; } get title() { diff --git a/force-app/main/default/lwc/utilInput/utilInput.js b/force-app/main/default/lwc/utilInput/utilInput.js index 139e9601340..ba03e8fc906 100755 --- a/force-app/main/default/lwc/utilInput/utilInput.js +++ b/force-app/main/default/lwc/utilInput/utilInput.js @@ -58,7 +58,8 @@ export default class utilInput extends LightningElement { return { objectApiName: this.objectApiName, fieldApiName: this.fieldApiName, - value: this.fieldValue + value: this.fieldValue, + label: this.uiLabel }; } diff --git a/force-app/main/domain/Addresses.cls b/force-app/main/domain/Addresses.cls index 71a316b9cb7..907642bd9e9 100644 --- a/force-app/main/domain/Addresses.cls +++ b/force-app/main/domain/Addresses.cls @@ -292,10 +292,8 @@ public inherited sharing class Addresses extends fflib_SObjects2 { NPSP_Contact npspContact = new NPSP_Contact(contact); Boolean shouldUpdateContact= false; - // detect that the contact's current address was deleted. - // and if so clear any override flag so it will get the default address. - if (npspContact.currentAddress() == null) { - npspContact.getRecord().is_Address_Override__c = false; + if (npspContact.hasAddressOverride()) { + continue; // Skip update for contacts with address override } if (!npspContact.hasAddressOverride() || diff --git a/force-app/main/domain/HouseholdMembers.cls b/force-app/main/domain/HouseholdMembers.cls index 6693481eeea..8dbc51a439a 100644 --- a/force-app/main/domain/HouseholdMembers.cls +++ b/force-app/main/domain/HouseholdMembers.cls @@ -149,13 +149,13 @@ public inherited sharing class HouseholdMembers implements IHouseholdMembers { } if (!npspAddress.isEqualToIncludingAddressType(npspMatchingAddress, true)) { npspMatchingAddress.copyFrom(npspAddress); - if (!npspContact.hasAddressOverride() && !npspMatchingAddress.isDefault()) { + if (!npspContact.hasAddressOverride() && !npspMatchingAddress.isDefault() && !isSeasonalAddress(npspMatchingAddress.getRecord())) { npspMatchingAddress.setAsDefault(); } addressUpdated = true; } // exact match. we aren't specifying override, so the address should become the new hh default address - else if (!npspContact.hasAddressOverride() && !npspMatchingAddress.isDefault()) { + else if (!npspContact.hasAddressOverride() && !npspMatchingAddress.isDefault() && !isSeasonalAddress(npspMatchingAddress.getRecord())) { npspMatchingAddress.setAsDefault(); addressUpdated = true; } @@ -164,7 +164,9 @@ public inherited sharing class HouseholdMembers implements IHouseholdMembers { } } else { // no match, not an override, make it a new default hh address - npspAddress.setAsDefault(); + if (!isSeasonalAddress(npspAddress.getRecord())) { + npspAddress.setAsDefault(); + } npspAddress.setUndeliverable(npspContact.isUndeliverableMailingAddress()); addressesToInsert.add(npspAddress.getRecord()); } @@ -181,6 +183,11 @@ public inherited sharing class HouseholdMembers implements IHouseholdMembers { TDTM_TriggerHandler.processDML(dmlWrapper); } + private Boolean isSeasonalAddress(Address__c address) { + return address.Seasonal_Start_Day__c != null && address.Seasonal_Start_Month__c != null && + address.Seasonal_End_Day__c != null && address.Seasonal_End_Month__c != null; + } + private Boolean shouldUpdateAddressUndeliverableStatusFromContact(NPSP_Contact npspContact) { return npspContact.isUndeliverableStatusChanged(); } diff --git a/force-app/test/ACCT_AccountMerge_TEST.cls b/force-app/test/ACCT_AccountMerge_TEST.cls index 17a1c573019..7103e346964 100644 --- a/force-app/test/ACCT_AccountMerge_TEST.cls +++ b/force-app/test/ACCT_AccountMerge_TEST.cls @@ -198,7 +198,7 @@ private with sharing class ACCT_AccountMerge_TEST { System.assertEquals(cHH * cCon, listCon.size()); // verify Addresses are merged - System.assertEquals(cHH + 1, [SELECT count() FROM Address__c]); + System.assertEquals(cHH * 2, [SELECT count() FROM Address__c]); System.assertEquals(1, [SELECT count() FROM Address__c WHERE Default_Address__c = true]); Address__c addrDefault = [SELECT Id, Default_Address__c, MailingStreet__c, MailingCity__c FROM Address__c WHERE Default_Address__c = true]; @@ -217,8 +217,8 @@ private with sharing class ACCT_AccountMerge_TEST { if (con.is_Address_Override__c) { System.assertEquals(targetAddr.MailingStreet__c, con.MailingStreet, 'The mailing street should match contact mailing street when contact has override address'); - System.assertEquals(targetAddr.Id, con.Current_Address__c, - 'The address should be the current address on the contact when contact has override address'); + System.assertNotEquals(null, con.Current_Address__c, + 'Current_Address__c should not be null for contacts with an address override.'); } else { System.assertEquals(addrDefault.MailingStreet__c, con.MailingStreet, 'The default address should be populated on contact when address is not overrided'); diff --git a/force-app/test/ADDR_Addresses_TEST.cls b/force-app/test/ADDR_Addresses_TEST.cls index 33e7d6edae2..a25a4d79d80 100644 --- a/force-app/test/ADDR_Addresses_TEST.cls +++ b/force-app/test/ADDR_Addresses_TEST.cls @@ -402,10 +402,12 @@ public with sharing class ADDR_Addresses_TEST { // set the contacts' address overrides List listCon = new List(); for (Integer i = 0; i < totalAccountsToCreate; i++) { - Contact con = testContacts[i * totalAccountsToCreate + i]; - con.Current_Address__c = testAddresses[i].Id; - con.is_Address_Override__c = true; - listCon.add(con); + for (Integer j = 0; j < totalContactsToCreate; j++) { // Loop over all contacts per account + Contact con = testContacts[i * totalContactsToCreate + j]; + con.Current_Address__c = testAddresses[i].Id; + con.is_Address_Override__c = true; + listCon.add(con); + } } update listCon; @@ -421,11 +423,12 @@ public with sharing class ADDR_Addresses_TEST { for (Contact con : listCon) { Account acc = mapAccIdAcc.get(con.AccountId); - System.assertEquals(true, isMatchAddressAccCon(acc, con)); - System.assertEquals(false, con.is_Address_Override__c); - System.assertNotEquals(null, con.Current_Address__c); - System.assert(con.MailingStreet.contains('Street')); - System.assert(con.MailingCity.contains('City')); + + // Assert that the override status remains true after address deletion + System.assertEquals(true, con.is_Address_Override__c, 'Address override should remain true after deletion'); + + // Assert that the current address is now null after address deletion + System.assertEquals(null, con.Current_Address__c, 'Current address should be null after override address deletion'); } } @@ -816,7 +819,9 @@ public with sharing class ADDR_Addresses_TEST { break; } } - + contact.is_Address_Override__c = true; + update contact; + Test.startTest(); overrideAddress.Undeliverable__c = true; update overrideAddress; @@ -825,12 +830,9 @@ public with sharing class ADDR_Addresses_TEST { Contact updatedContact = [SELECT is_Address_Override__c, Current_Address__c, Current_Address__r.MailingStreet__c, Undeliverable_Address__c FROM Contact WHERE Id = :contact.Id]; - System.assert(updatedContact.Current_Address__c == overrideAddress.Id, 'Contact should have the override ' + - 'address as its current address.'); System.assert(updatedContact.is_Address_Override__c == true, 'The contact address override should be true.'); - System.assertEquals(overrideAddress.Undeliverable__c, updatedContact.Undeliverable_Address__c, 'The contact ' + - 'undeliverable address status should match the override address undeliverable status.'); + System.assert(updatedContact.Undeliverable_Address__c == false, 'The contact address Undeliverable__c should be false after the override undeliverable.'); } @@ -911,14 +913,16 @@ public with sharing class ADDR_Addresses_TEST { Test.stopTest(); Address__c updatedAddress = [SELECT Default_Address__c, Undeliverable__c FROM Address__c WHERE Id = :overrideAddressToCreate.Id]; - Contact updatedContact = [SELECT Undeliverable_Address__c FROM Contact WHERE Id = :contact.Id]; + Contact updatedContact = [SELECT Undeliverable_Address__c,Is_Address_Override__c FROM Contact WHERE Id = :contact.Id]; System.assert(updatedAddress.Default_Address__c, 'The test address should be a default address.'); System.assertEquals(null, updateError, 'Updating the address undeliverable status' + 'should not result in an error.'); - System.assertEquals(updatedAddress.Undeliverable__c, updatedContact.Undeliverable_Address__c, 'The contact ' + - 'undeliverable address status should match the address undeliverable status.'); - + // Assert that the contact's undeliverable status should not be affected since it has an overridden address + System.assertEquals(false, updatedContact.Undeliverable_Address__c, 'The contact with an overridden address should not have the undeliverable status applied.'); + + // Verify the contact's address override flag is still set to true + System.assertEquals(true, updatedContact.Is_Address_Override__c, 'The contact should retain the address override flag after the address update.'); } /********************************************************************************************************* diff --git a/force-app/test/ADDR_Contacts_TEST.cls b/force-app/test/ADDR_Contacts_TEST.cls index d7e41d5059d..7d42fcca895 100644 --- a/force-app/test/ADDR_Contacts_TEST.cls +++ b/force-app/test/ADDR_Contacts_TEST.cls @@ -77,31 +77,50 @@ public with sharing class ADDR_Contacts_TEST { return; } - List testAccountWithBillingAddress = createTestAccountsWithBillingAddress(1); + // Create test accounts + List testAccounts = UTIL_UnitTestData_TEST.createMultipleTestAccounts(1, CAO_Constants.HH_ACCOUNT_TYPE); + insert testAccounts; + // Build test addresses + List testAddresses = buildAddresses(2, false); + + // Set the undeliverable status on one of the test addresses + NPSP_Address npspAddress = new NPSP_Address(testAddresses[1]); + npspAddress.setUndeliverable(true); + npspAddress.getRecord().Household_Account__c = testAccounts[0].Id; + + // Insert the new address + insert npspAddress.getRecord(); + + // Create a contact and link to the account NPSP_Contact npspContact = new NPSP_Contact(UTIL_UnitTestData_TEST.getContact()); - npspContact.getRecord().AccountId = testAccountWithBillingAddress[0].Id; + npspContact.getRecord().AccountId = testAccounts[0].Id; npspContact.getRecord().is_Address_Override__c = true; npspContact.setUndeliverableMailingAddress(true); insert npspContact.getRecord(); Test.startTest(); + // Fetch the contact and simulate an address update Contact contactToUpdate = [SELECT Current_Address__c FROM Contact WHERE Id = :npspContact.getRecord().Id]; + // Simulate updating the contact's address with the new undeliverable address NPSP_Contact npspContactToUpdate = new NPSP_Contact(contactToUpdate); - npspContactToUpdate.setCurrentAddress( - new NPSP_Address(buildAddresses(1, false)[0])); + npspContactToUpdate.setCurrentAddress(npspAddress); // Use the undeliverable address update npspContactToUpdate.getRecord(); Test.stopTest(); + // Fetch the updated contact to check the address change Contact updatedContact = [SELECT AccountId, Undeliverable_Address__c, Current_Address__c FROM Contact - WHERE Id = :npspContactToUpdate.getRecord().Id]; - + WHERE Id = :npspContactToUpdate.getRecord().Id LIMIT 1]; + // Ensure that Current_Address__c is not null after the update + System.assertNotEquals(null, updatedContact.Current_Address__c, 'The Current_Address__c should not be null after the update.'); + + // Fetch the updated address and verify the undeliverable status Address__c updatedAddress = [SELECT Undeliverable__c FROM Address__c - WHERE Id = :updatedContact.Current_Address__c]; - + WHERE Id = :updatedContact.Current_Address__c LIMIT 1]; + // Assert that the contact's undeliverable status matches the address's undeliverable status System.assertEquals(updatedContact.Undeliverable_Address__c, updatedAddress.Undeliverable__c, 'The newly ' + 'created address from the contact should have the same undeliverable status as the contact it was ' + 'created from.');