Skip to main content
Switch Language
  • Software

WERCS Studio

Chemical Data and Compliance Management

Achieve regulatory compliance, supply chain transparency and meet your sustainability initiatives with WERCS Studio software.

Line of colored rectangles with two selected Illustration of a chemical structure next to a warning symbol and a checkmark

Chemical compliance management and enterprise SDS authoring software

Looking for an easier, more powerful way to manage chemical and material data from development through disposal to meet regulatory requirements, create product transparency and enable safer products?

WERCS Studio is an industry-leading software solution that makes it easy to automate and centralize compliance data — an essential part of ULTRUS™ software from UL Solutions that helps you accelerate your product stewardship needs.

This software is powered by comprehensive ChemADVISOR® regulatory data that helps you efficiently manage and automate your environment, health and safety (EHS) initiatives and mitigate compliance risks.

 

Comprehensive chemical data management

 

Icon of arrows in a circle

Product stewardship

Manage the full scope of your product stewardship journey from sourcing of materials to product disposal.  

Icon of a building with columns

Regulatory compliance

Help meet your regulatory compliance obligations to mitigate risk and enable global market access.

Icon of filtering data

Material and product data

Access comprehensive and insightful product and material data for improved decision-making and supply chain transparency.

Icon of a document with a stamp next to it

Policy management

Establish and enforce your chemical management policies.  

 

Leading chemical compliance management software

Flexible and scalable, WERCS Studio can help you manage the chemicals in your products throughout the supply chain. Solve your everyday product stewardship challenges with:

  • Automated Safety Data Sheet (SDS) authoring and labeling
  • Worldwide compliance with the Globally Harmonized System (GHS)
  • Integrated chemical regulatory content database
  • More than 8,000 global regulatory lists
  • More than 25,000 safety phrases
  • More than 45 languages supported with translated phrases
  • More than 30 scalable modules
  • SDS authoring and labeling templates for any region managed by UL Solutions
Schedule a demo
More than 25,000 certified safety phases
More than 8,000 global regulatory lists
More than 45 languages supported with translated phrases
More than 30 scalable modules

 

 

Take control of product stewardship and chemical data management

 

 

Icon of a data model

Manage and distribute data

Intelligently share, analyze and manage your product-related data. Our powerful digital platform can help you connect with your supply chain.

Icon of software interface

Author SDSs and manage chemical and material data

Assess materials and create and distribute compliance documents for your finished products.

Icon of a mouse clicking a globe

Access current global chemical regulations

Our world-renowned ChemADVISOR regulatory database allows you to quickly access up-to-date and comprehensive chemical regulatory data and intelligence.

 

Icon of a data report

Customized chemical policy management

Create and manage a customized chemical policy based on your unique needs to effectively manage the materials used in your products.

Icon of a circle with six circles branching off of it

Configurable, with integration capabilities

Our robust chemical and material data management software can integrate with your internal systems and is scalable to help meet your compliance needs — for now and into the future.

Icon of three people in a triangle formation inside a shield

Backed by world-class regulatory expertise

Our global team of more than 70 experienced regulatory experts uses the software every day to work with organizations across many industries to assist them with chemical data management and compliance needs.

 

WERCS Studio supports regulatory compliance and EHS management across a diverse set of industries:

  • Industrial chemicals
  • Petroleum
  • Paints and coatings
  • Fragrances and flavors
  • Lubricants
  • Agrochemicals
  • Consumer product goods
  • Electrical and electronic equipment (EEE)
  • Wire and cable
  • Retail
  • Pharmaceuticals
  • Animal health

 

 

Ready to elevate your chemical data management compliance program?

Schedule a complimentary demo today to learn how you can leverage WERCS Studio to help you easily and efficiently manage your data, meet regulatory compliance obligations and enable supply chain transparency.

Schedule a demo

 

Regulatory Roundup Newsletter

Stay up to date on chemical regulatory news

Subscribe to our monthly Regulatory Roundup Newsletter and stay up to date on current and upcoming regulations and all the latest chemical industry news. 

Subscribe now
X

Ready to elevate your chemical data management compliance program?

Schedule a complimentary demo today to learn how you can leverage WERCS Studio to help you easily and efficiently manage your data, meet regulatory compliance obligations and enable supply chain transparency.

Please wait…
'; let submitButton = $('form.mktoForm .mktoButtonWrap .mktoButton'); submitButton.prop("disabled", true); submitButton.append(loadingHTML); // Ocp-Apim-Subscription-Key: '***'. var $ocpKey = drupalSettings.ul_marketo_validate_key; var $ocpUrl = drupalSettings.ul_marketo_validate_url; var $ocpEnv = drupalSettings.ul_marketo_validate_env; if (submitCount > 1 || isResponseSuccess == true) { isResponseHandled = true; form.submittable(true); console.log("NO.10 submitCount > 1: form.submit() "); submitButton.click(); } else { var email = form.vals().Email; var phone = (form.vals().Phone) ? (form.vals().Phone) : '18472728800'; var country = (form.vals().Country) ? (form.vals().Country) : 'United States'; var countryCode = (phone == '18472728800') ? 'US' : getCountryCode(country); var $emailMsg = drupalSettings.ul_marketo_validate.email; var $phoneMsg = drupalSettings.ul_marketo_validate.phone; console.log("paramsDebug:", paramsDebug); sendingData = JSON.stringify({ 'Input_Email': email, 'Input_Phone': phone, 'Input_ISO2_Country_Code': countryCode }); console.log('URL, debug missing button:', $ocpUrl); console.log("Sending Data to API service (ajax):", sendingData); // API call for validate email/phone/country. var sendDate = (new Date()).getTime(); var responseTimeMs = sendDate; // Start a timer to handle a 3-second response timeout setTimeout(function() { if (!isResponseHandled) { isResponseHandled = true; // Handle the assumed "success" due to timeout here console.log('No response within timeout, proceeding with default validation success.'); $('#ValidMsgEmail').remove(); $('#ValidMsgPhone').remove(); // set form to be submittable here $('#valSpinner').remove(); submitButton.prop("disabled", false); // Setup status as Timeout instead of Unknown. emailAddressStatus = "Timeout"; phoneNumberStatus = "Timeout"; phoneNumberValidated = "Timeout"; form.vals({ "emailAddressStatus": emailAddressStatus, "phoneNumberStatus": phoneNumberStatus, "phoneNumberValidated": phoneNumberValidated, }); form.submittable(true); isResponseSuccess = true; let receiveDate = (new Date()).getTime(); responseTimeMs = (receiveDate - sendDate) / 1000; console.log("setTimeout: " + responseTimeMs + " seconds."); submitButton.click(); } }, paramsDebug.timeout * 1000); //END setTimeout $.ajax({ url: $ocpUrl, type: 'POST', data: sendingData, headers: { 'Content-Type': 'application/json', 'Ocp-Apim-Subscription-Key': $ocpKey, }, success: function (response) { $('#valSpinner').remove(); submitButton.prop("disabled", false); phoneNumberValidated = response.Validated_Phone_Number; if (!isResponseHandled) { isResponseHandled = true; // **** Handle the validation response console.log("API Response Data: ", response); var emailCode = response.Email_Validation_Status_Number; var phoneCode = response.Phone_Validation_Status_Number; // Add values into two status fields. if (emailValidCode.hasOwnProperty(emailCode)) { emailAddressStatus = emailValidCode[emailCode]; } else { // API error code doesn't exist, set defaul. emailAddressStatus = emailValidCode['300']; } if (phoneValidCode.hasOwnProperty(phoneCode)) { phoneNumberStatus = phoneValidCode[phoneCode]; } else { // API error code doesn't exist, set defaul. phoneNumberStatus = phoneValidCode['301']; } if (emailAddressStatus.indexOf("email_not") != -1) { emailAddressStatus = 'Invalid'; } else if(emailAddressStatus == 'valid') { emailAddressStatus = 'Valid'; } if (phoneNumberStatus.indexOf('invalid_') != -1) { phoneNumberStatus = 'Invalid'; } else if(phoneNumberStatus == 'valid') { phoneNumberStatus = 'Valid'; } console.log('form.emailAddressStatus ' + emailAddressStatus); console.log('form.phoneNumberStatus ' + phoneNumberStatus); console.log('form.phoneNumberValidated ' + phoneNumberValidated); if (validEmailCode.includes(emailCode) && validPhoneCode.includes(phoneCode)) { // Remove the loading spinner. $('#ValidMsgEmail').remove(); $('#ValidMsgPhone').remove(); $('#valSpinner').remove(); submitButton.prop("disabled", false); console.log("NO.1.1 VALID OK submitCount = " + submitCount); isResponseSuccess = true; form.vals({ "emailAddressStatus": emailAddressStatus, "phoneNumberStatus": phoneNumberStatus, "phoneNumberValidated": phoneNumberValidated, }); form.submittable(true); submitButton.click(); } // API valid code is not "valid". else { form.submittable(false); console.log("NO.1.2 Submittable false : count= " + submitCount + " validateOrigin " + validateOrigin + " isResponseSuccess " + isResponseSuccess ); // Set the invalid message in language translation. var msgEmailStatus = ""; if (!validEmailCode.includes(emailCode)) { if (emailCode == '300' || emailCode == '400') { msgEmailStatus = $emailMsg.email_not_valid; } else if (emailCode == '310' || emailCode == '500' ) { msgEmailStatus = $emailMsg.email_not_accept; } else { msgEmailStatus = "Unknown"; } console.log("NOT validEmailCode: " + msgEmailStatus); errEmail = ''; $('#Email').after(errEmail); form.submittable(false); } // Set the invalid message in language translation. var msgPhoneStatus = "" if (!validPhoneCode.includes(phoneCode)) { msgPhoneStatus = $phoneMsg[phoneValidCode[phoneCode]]; console.log("NOT validPhoneCode: " + msgPhoneStatus); errPhone = ''; $('#Phone').after(errPhone); form.submittable(false); } console.log('msgEmailStatus:', msgEmailStatus); console.log('msgPhoneStatus:', msgPhoneStatus); // Remove the loading spinner. $('#valSpinner').remove(); submitButton.prop("disabled", false); console.log("NO.1.6 Submittable false, count= " + submitCount); form.vals({ "emailAddressStatus": emailAddressStatus, "phoneNumberStatus": phoneNumberStatus, "phoneNumberValidated": phoneNumberValidated, }); // 2nd API call and subit form. if (submitCount >= 1) { $('#ValidMsgEmail').remove(); $('#ValidMsgPhone').remove(); isResponseSuccess = true; form.submittable(true); console.log("NO.1.8 : 2nd Submit:: submittable=true && count=1 : " + submitCount); if (submitCount==1) { submitButton.click(); } } } //END if{} else{}. }//END if (!isResponseHandled) // Calculate the time comsumed for the API call. let receiveDate = (new Date()).getTime(); responseTimeMs = (receiveDate - sendDate) / 1000; console.log("NO.1.9 : AJAX success: Time for API call: " + responseTimeMs + " seconds."); submitCount++; }, //END success: function(); // API call error response. error: function (error) { $('#valSpinner').remove(); submitButton.prop("disabled", false); if (!isResponseHandled) { isResponseHandled = true; $('#valSpinner').remove(); submitButton.prop("disabled", false); // 2nd API call and subit form. if ( submitCount >= 1 ) { $('#ValidMsgEmail').remove(); $('#ValidMsgPhone').remove(); form.submittable(true); isResponseSuccess = true; if ( submitCount == 1 ) { form.submittable(true); } } // Handle AJAX error console.log('**** 2 ajax error. submitCount = ' + submitCount); console.log(error); // Calculate the time comsumed for the API call. let receiveDate = (new Date()).getTime(); responseTimeMs = (receiveDate - sendDate) / 1000; console.log("NO.2.3 AJAX error: Time for API call: " + responseTimeMs + " seconds."); } submitCount++; // Error status 500, then submit the form. submitButton.click(); } //END error: function(); }); //END $.ajax; if (isResponseSuccess) { console.log("NO.8.0 submittable(true)"); form.submittable(true); } } //END: if (submitCount > 1) else // Setup the form.vals and form.submittable; form.vals({ "emailAddressStatus": emailAddressStatus, "phoneNumberStatus": phoneNumberStatus, "phoneNumberValidated": phoneNumberValidated, }); } //END: if (validateOrigin && (marketoBundle)) else { // For Event form and Newsletter form. if (validateOrigin === true){ form.submittable(true); } } }); //END form.onValidate() // Success callback() form.onSuccess(function(values, followUpUrl){ // Debug Phone/Email validation. console.log("NO.9.1 onSuccess: submittable = " + form.submittable()); console.log(form.getValues()); // Track analytics. if (typeof dataLayer !== 'undefined'){ dataLayer.push({ event: drplMkto.dataLayerEvent, mktoFormId: form.getId(), 'FormValues': cleanFormVals(form.getValues()), 'FormFields': form.allFieldsFilled(), 'Submittable': form.submittable() }); } // If function exists, delete UTM cookie: if(typeof _deleteUtmCookie === "function"){ _deleteUtmCookie(); } // Marketo Modal "Thank You" message: if(use_post_submit_mssg){ $('html, body').animate({ scrollTop: 0 }, 'slow'); $('.mkto-presubmit').addClass('hidden'); $('.mktoModalContent').addClass('mkto_thnx_center'); $('.mktoModalMask').addClass('not_clicable'); $('.mkto-postsubmit').removeClass('hidden'); $('.mktoButton').removeAttr('disabled').text(drupalSettings.marketo.button_text); $('.mktoForm')[0].reset(); grecaptcha.reset(); } // Else, redirect user: else { window.location.href = drplMkto.success_url; } // IMPORTANT: Return false to prevent further code execution. return false; }); //END form.onSuccess. }); })(jQuery, drupalSettings.marketo);