It's simple enough to create a utility to enforce that string literals are lowercase using the type utility Lowercase<StringType>:
TS Playground
type EnforceLowerCase<T extends string> = Lowercase<T> extends T ?
string extends T ? never : T
: never;
type Str1 = EnforceLowerCase<'Hello'>; // never
type Str2 = EnforceLowerCase<'hello'>; // "hello"
However, using it in your scenario would be very convoluted, because it would require using only string literals (which TS doesn't infer on object property values).
Instead, it's better to ensure conformance of the expectation at runtime. Here's an example using a pure function which doesn't mutate the input object:
TS Playground
function ensureLowerCaseFieldName (props: Props): Props {
return {...props, fields: {
...props.fields,
fieldName: props.fields.fieldName.toLocaleLowerCase(),
}};
}
function doSomethingWithProps (props: Props) {
props = ensureLowerCaseFieldName(props);
console.log(props);
}
const props: Props = {
onEdit: data => {},
title: 'title',
fields: {fieldName: 'Hello'},
};
doSomethingWithProps(props); // {..., fields: { fieldName: "hello" } }
|, documentation here : typescriptlang.org/docs/handbook/2/template-literal-types.htmlfieldName.toLowerCase() === fieldName. Or write 30 lines of overkill typing nonsense.