-
Notifications
You must be signed in to change notification settings - Fork 12.6k
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
Template strings: Negated match #49867
Comments
If we presuppose type Name = `${not "--"}${string}`; which we can parse only in the context of a |
@RyanCavanaugh That was my initial idea, but presumably the type So given the string
Negating an entire pattern is less ambiguous: string & not `--${string}` |
I think the empty string matching argument is correct under one interpretation of how that should work. I guess the problem is that if Is it as simple as for I feel like we have to handle it at a placeholder level since the template strings can already be nested, so we'd have to have answers for what it means to write type Part = not `${'a' | 'b'}`;
type Whole = `${Part}${Part}${Part}`; |
Maybe someone else does, but I don’t have any particularly more complicated cases than the one I mentioned. So I would accept a solution that simply doesn’t allow you to interpolate negated strings into other template types. type A = not "a"
type B = `${A}` // invalid |
w.r.t. the empty string issue, isn't it already the case that when there are consecutive placeholders, each one except for the last must consume at least one character (in order to enable recursive |
Consecutive placeholders will consume one character at a time if present, but will still produce type Three = `${string}${string}${string}`;
const a: Three = "mm"; is considered to be effectively |
Right, but the empty string can only be produced by the final one unless there aren't enough characters to begin with. That's why I said "except for the last" above. In other words, |
Thinking about this more, I wonder if a lot of the concerns about the original type NonEmptyString = string & not ""
type NonZeroNumber = number & not 0
type NonSymbolPropertyKey = PropertyKey & not symbol interface Nums {
label: string
[index: string & not "label"]: number
} interface DesignTokens {
$type: string
$description: string
[index: string & not `$${string}`]: any
}
From the original pull request (#29317), from what I can gather, these were the primary concerns: (cc @DanielRosenwasser)
// (this is an error because objects in TS are not 'exact')
let o: not { bar: string } = { foo: string } // Error!
Flexibility around primitive and literal types is the main use case that I'm concerned with, and I think I would actually prefer if TypeScript told me when I was trying to do silly stuff like |
Why not just allow regex types instead of re-inventing the wheel of regex? There are plenty of use cases. For example, creating a type that only allows a valid CSS hex color as a value is extremely non-trivial. |
@matthew-dean : Regex would be fricken sensational. there is a library for this but performance may be problematic: |
Suggestion
📃 Motivating Example
Strings, in particular key names, sometimes affect types based on the format of the string, most commonly with prefixes. One such example is the recent W3C Design Tokens spec which uses a
$
prefix to reserve key names like$description
and$type
.Right now it's possible to create a positive match such as:
But there's no way to create a negative match. Or in regex terms:
The goal here wouldn't be to recreate all the functionality of regex/parsing, it would be to handle the stringy-typing sometimes seen in JavaScript (including within official web specifications).
⭐ Suggestion
Expose, in a limited capacity, the
not
operator (#29317) so that it can be used to filter strings.There is an existing PR adding a
not
operator, but has been long stalled on expected behavior. But maybe just this one piece could be added and slowly expanded from, if desired, later.💻 Use Cases
The W3C Design Tokens spec does not allow token/group names starting with a
$
or containing any of the characters.{}
.Using negated string matches, you could correctly type this:
The closest you can get to this today is:
🔍 Search Terms
List of keywords you searched for before creating this issue. Write them down here so that others can find this suggestion more easily and help provide feedback.
✅ Viability Checklist
My suggestion meets these guidelines:
The text was updated successfully, but these errors were encountered: