Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Cancel Original Data State (oldData) Update From Built-In Save Action #125

Open
TimTraylor opened this issue Mar 22, 2022 · 6 comments
Open

Comments

@TimTraylor
Copy link

Hi Luca & team,

Nice work on all of this!

I'm wondering if there is a way to stop the automatic updating of the oldData to the formData from the built-in 'Save' action. As an example, I'm passing in a form-level validation function along with the schema, and I execute this additional function before saving to the back-end. If that validation fails I'll exit the function (defined on the Save button) without saving, the idea being that they need to continue to edit the form.

While I can force the form back into edit mode, at that point it has already made the 'oldData' equal to the 'formData', so if you click 'Cancel' after that it doesn't change it back to the original.

Is there any way to cancel the default behavior and keep them in edit without having the oldData get updated?

Thanks!
Tim

@mesqueeb
Copy link
Member

mesqueeb commented Mar 22, 2022

This should be the case when using the built in validation system. Or is using this not an option for you?

Can you show me some code so I can think about the best solution?

@TimTraylor
Copy link
Author

TimTraylor commented Mar 23, 2022

I was trying to address cross-field validation. So things like "If 'ACH' is chosen as a payment method, then checking account and transit routing numbers are required". Or generically, "If the value of 'foo' is A, B, or C then the value entered for 'bar' must be between 5 and 10'. Those types of things.

So, what's being returned from an API call would look something like this (most schema fields removed for brevity):

const apiResult = {
    schema: [ 
    {
      id: "accountNumber",
      component: "input",
      label: "Account number",
      required: true,
      dynamicProps: ['error'],
      error: (val) => (val != null && val.length == 10 ? null : 'Must be 10 characters'),
    },
   {
      id: "transitRouting",
      component: "input",
      label: "Transit Routing",
      dynamicProps: ['disabled'],
      disabled: (val, { formData }) => formData.paymentType != 'ach',
    },
  ...],
   validationFunctions: [
      function() {
        if (formData.value.paymentType == 'ach') {
            if (!formData.value.transitRouting) {
                toaster.show('Transit routing is required for ACH.');
                return false;
            }
            if (!Number(formData.value.transitRouting) || 
                formData.value.transitRouting.length != 9) {
                toaster.show('Transit routing must be numeric and 9 digits.');
                return false;
            }
        }
        return true;
      },
    ]
  };

And then the built-in save action button calls a function like this:

function onSave(payload){
  
  let passedValidation = true;

  apiResult.validationFunctions.forEach(    
      function(func) {
        if (!func())
        {
          passedValidation = false;
          mode.value='edit';
          return;
        }
    }
  );
  
  if (!passedValidation){
    return;
  }

  toaster.show(`Would save the payload:` + JSON.stringify(payload, null, 2));
  clearForm();
  forceRemount.value++;
}

I'm doing a POC here, hence the showing of the error with toasters. The real thing would appropriately display the error on the page, styled as the built-in Blitzar errors are.

I think you can see though that I'm using the built-in field-level validation for the values of the individual items, but then I'm also trying to implement a form-level process that can address cross-field validation.

Thanks,
Tim

@mesqueeb
Copy link
Member

mesqueeb commented Apr 5, 2022

@TimTraylor can you add me on discord:
Luca Ban [Mesqueeb]#4774

@mesqueeb
Copy link
Member

mesqueeb commented Apr 5, 2022

@TimTraylor I tried to show you an example using dynamicProps for the error that would achieve something similar to what you showed me:

{
  id: "transitRouting",
  component: "input",
  label: "Transit Routing",
  dynamicProps: ['disabled', 'error'],
  disabled: (val, { formData }) => formData.paymentType != 'ach',
  error: (val, { formData }) => {
    return formData.paymentType === 'ach' && !formData.transitRouting
      ? 'Transit routing is required for ACH.'
      : null
  }
},

just one example, but I can imagine that this way of writing your validation functions should be usable for the other things you wanted to achieve as well?

@TimTraylor
Copy link
Author

Thanks Luca, that was just a simple use case of cross-field validation at the form level, but the idea was needing something to execute at the form level before allowing a submit to be 'successful'. If you have any complex multi-field validation, trying to attach that to just one of the fields can be difficult.

I could certainly do it without using the built-in action buttons, but I really liked their ease of use.

Thanks for replying - I appreciate the response!

@mesqueeb
Copy link
Member

mesqueeb commented Apr 9, 2022

I can introduce a form level error prop that in checked when clicking the save action button.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

2 participants