Skip to content

Commit bd4b701

Browse files
committedJan 3, 2024
feat: converting from starting php repo
1 parent 45c6b06 commit bd4b701

33 files changed

+1113
-0
lines changed
 

‎README.md

+2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,8 @@
11
# tailwind-merge-nuget
22
A TailwindCSS class merge utility similar to Shadcn UI / Tailwind Merge
33

4+
https://github.com/YieldStudio/tailwind-merge-php/tree/main
5+
46
**NOTE** this is a __very__ simple implementation of Tailwind class merging and likely has many, many edge cases that are not covered. It is a simple implementation that works for my current use case. If you find a bug, please submit a PR with a test case and I will strive to update this package.
57

68
## Usage

‎TailwindMerge.Tests/ValidatorTests.cs

+199
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,199 @@
1+
namespace TailwindMerge.Tests;
2+
using TailwindMerge; // Replace with the namespace of your class library
3+
4+
public class ValidatorTests
5+
{
6+
[Fact]
7+
public void IsLengthTests()
8+
{
9+
Assert.True(Validators.IsLength("1"));
10+
Assert.True(Validators.IsLength("1023713"));
11+
Assert.True(Validators.IsLength("1.5"));
12+
Assert.True(Validators.IsLength("1231.503761"));
13+
Assert.True(Validators.IsLength("px"));
14+
Assert.True(Validators.IsLength("full"));
15+
Assert.True(Validators.IsLength("screen"));
16+
Assert.True(Validators.IsLength("1/2"));
17+
Assert.True(Validators.IsLength("123/345"));
18+
19+
Assert.False(Validators.IsLength("[3.7%]"));
20+
Assert.False(Validators.IsLength("[481px]"));
21+
Assert.False(Validators.IsLength("[19.1rem]"));
22+
Assert.False(Validators.IsLength("[50vw]"));
23+
Assert.False(Validators.IsLength("[56vh]"));
24+
Assert.False(Validators.IsLength("[length:var(--arbitrary)]"));
25+
Assert.False(Validators.IsLength("1d5"));
26+
Assert.False(Validators.IsLength("[1]"));
27+
Assert.False(Validators.IsLength("[12px"));
28+
Assert.False(Validators.IsLength("12px]"));
29+
Assert.False(Validators.IsLength("one"));
30+
}
31+
32+
[Fact]
33+
public void IsArbitraryLengthTests()
34+
{
35+
Assert.True(Validators.IsArbitraryLength("[3.7%]"));
36+
Assert.True(Validators.IsArbitraryLength("[481px]"));
37+
Assert.True(Validators.IsArbitraryLength("[19.1rem]"));
38+
Assert.True(Validators.IsArbitraryLength("[50vw]"));
39+
Assert.True(Validators.IsArbitraryLength("[56vh]"));
40+
Assert.True(Validators.IsArbitraryLength("[length:var(--arbitrary)]"));
41+
42+
Assert.False(Validators.IsArbitraryLength("1"));
43+
Assert.False(Validators.IsArbitraryLength("3px"));
44+
Assert.False(Validators.IsArbitraryLength("1d5"));
45+
Assert.False(Validators.IsArbitraryLength("[1]"));
46+
Assert.False(Validators.IsArbitraryLength("[12px"));
47+
Assert.False(Validators.IsArbitraryLength("12px]"));
48+
Assert.False(Validators.IsArbitraryLength("one"));
49+
}
50+
51+
[Fact]
52+
public void IsIntegerTests()
53+
{
54+
Assert.True(Validators.IsInteger("1"));
55+
Assert.True(Validators.IsInteger("123"));
56+
Assert.True(Validators.IsInteger("8312"));
57+
58+
Assert.False(Validators.IsInteger("[8312]"));
59+
Assert.False(Validators.IsInteger("[2]"));
60+
Assert.False(Validators.IsInteger("[8312px]"));
61+
Assert.False(Validators.IsInteger("[8312%]"));
62+
Assert.False(Validators.IsInteger("[8312rem]"));
63+
Assert.False(Validators.IsInteger("8312.2"));
64+
Assert.False(Validators.IsInteger("1.2"));
65+
Assert.False(Validators.IsInteger("one"));
66+
Assert.False(Validators.IsInteger("1/2"));
67+
Assert.False(Validators.IsInteger("1%"));
68+
Assert.False(Validators.IsInteger("1px"));
69+
}
70+
71+
[Fact]
72+
public void IsArbitraryValueTests()
73+
{
74+
Assert.True(Validators.IsArbitraryValue("[1]"));
75+
Assert.True(Validators.IsArbitraryValue("[bla]"));
76+
Assert.True(Validators.IsArbitraryValue("[not-an-arbitrary-value?]"));
77+
Assert.True(Validators.IsArbitraryValue("[auto,auto,minmax(0,1fr),calc(100vw-50%)]"));
78+
79+
Assert.False(Validators.IsArbitraryValue("[]"));
80+
Assert.False(Validators.IsArbitraryValue("[1"));
81+
Assert.False(Validators.IsArbitraryValue("1]"));
82+
Assert.False(Validators.IsArbitraryValue("1"));
83+
Assert.False(Validators.IsArbitraryValue("one"));
84+
Assert.False(Validators.IsArbitraryValue("o[n]e"));
85+
}
86+
87+
[Fact]
88+
public void IsAnyTests()
89+
{
90+
Assert.True(Validators.IsAny());
91+
// TypeScript specific @ts-expect-error tests are not applicable in C#
92+
}
93+
94+
[Fact]
95+
public void IsTshirtSizeTests()
96+
{
97+
Assert.True(Validators.IsTshirtSize("xs"));
98+
Assert.True(Validators.IsTshirtSize("sm"));
99+
Assert.True(Validators.IsTshirtSize("md"));
100+
Assert.True(Validators.IsTshirtSize("lg"));
101+
Assert.True(Validators.IsTshirtSize("xl"));
102+
Assert.True(Validators.IsTshirtSize("2xl"));
103+
Assert.True(Validators.IsTshirtSize("2.5xl"));
104+
Assert.True(Validators.IsTshirtSize("10xl"));
105+
Assert.True(Validators.IsTshirtSize("2xs"));
106+
Assert.True(Validators.IsTshirtSize("2lg"));
107+
108+
Assert.False(Validators.IsTshirtSize(""));
109+
Assert.False(Validators.IsTshirtSize("hello"));
110+
Assert.False(Validators.IsTshirtSize("1"));
111+
Assert.False(Validators.IsTshirtSize("xl3"));
112+
Assert.False(Validators.IsTshirtSize("2xl3"));
113+
Assert.False(Validators.IsTshirtSize("-xl"));
114+
Assert.False(Validators.IsTshirtSize("[sm]"));
115+
}
116+
117+
[Fact]
118+
public void IsArbitrarySizeTests()
119+
{
120+
Assert.True(Validators.IsArbitrarySize("[size:2px]"));
121+
Assert.True(Validators.IsArbitrarySize("[size:bla]"));
122+
Assert.True(Validators.IsArbitrarySize("[length:bla]"));
123+
Assert.True(Validators.IsArbitrarySize("[percentage:bla]"));
124+
125+
Assert.False(Validators.IsArbitrarySize("[2px]"));
126+
Assert.False(Validators.IsArbitrarySize("[bla]"));
127+
Assert.False(Validators.IsArbitrarySize("size:2px"));
128+
}
129+
130+
[Fact]
131+
public void IsArbitraryPositionTests()
132+
{
133+
Assert.True(Validators.IsArbitraryPosition("[position:2px]"));
134+
Assert.True(Validators.IsArbitraryPosition("[position:bla]"));
135+
136+
Assert.False(Validators.IsArbitraryPosition("[2px]"));
137+
Assert.False(Validators.IsArbitraryPosition("[bla]"));
138+
Assert.False(Validators.IsArbitraryPosition("position:2px"));
139+
}
140+
141+
[Fact]
142+
public void IsArbitraryImageTests()
143+
{
144+
Assert.True(Validators.IsArbitraryImage("[url:var(--my-url)]"));
145+
Assert.True(Validators.IsArbitraryImage("[url(something)]"));
146+
Assert.True(Validators.IsArbitraryImage("[url:bla]"));
147+
Assert.True(Validators.IsArbitraryImage("[image:bla]"));
148+
Assert.True(Validators.IsArbitraryImage("[linear-gradient(something)]"));
149+
Assert.True(Validators.IsArbitraryImage("[repeating-conic-gradient(something)]"));
150+
151+
Assert.False(Validators.IsArbitraryImage("[var(--my-url)]"));
152+
Assert.False(Validators.IsArbitraryImage("[bla]"));
153+
Assert.False(Validators.IsArbitraryImage("url:2px"));
154+
Assert.False(Validators.IsArbitraryImage("url(2px)"));
155+
}
156+
157+
[Fact]
158+
public void IsArbitraryNumberTests()
159+
{
160+
Assert.True(Validators.IsArbitraryNumber("[number:black]"));
161+
Assert.True(Validators.IsArbitraryNumber("[number:bla]"));
162+
Assert.True(Validators.IsArbitraryNumber("[number:230]"));
163+
Assert.True(Validators.IsArbitraryNumber("[450]"));
164+
165+
Assert.False(Validators.IsArbitraryNumber("[2px]"));
166+
Assert.False(Validators.IsArbitraryNumber("[bla]"));
167+
Assert.False(Validators.IsArbitraryNumber("[black]"));
168+
Assert.False(Validators.IsArbitraryNumber("black"));
169+
Assert.False(Validators.IsArbitraryNumber("450"));
170+
}
171+
172+
[Fact]
173+
public void IsArbitraryShadowTests()
174+
{
175+
Assert.True(Validators.IsArbitraryShadow("[0_35px_60px_-15px_rgba(0,0,0,0.3)]"));
176+
Assert.True(Validators.IsArbitraryShadow("[0_0_#00f]"));
177+
Assert.True(Validators.IsArbitraryShadow("[.5rem_0_rgba(5,5,5,5)]"));
178+
Assert.True(Validators.IsArbitraryShadow("[-.5rem_0_#123456]"));
179+
Assert.True(Validators.IsArbitraryShadow("[0.5rem_-0_#123456]"));
180+
Assert.True(Validators.IsArbitraryShadow("[0.5rem_-0.005vh_#123456]"));
181+
Assert.True(Validators.IsArbitraryShadow("[0.5rem_-0.005vh]"));
182+
183+
Assert.False(Validators.IsArbitraryShadow("[rgba(5,5,5,5)]"));
184+
Assert.False(Validators.IsArbitraryShadow("[#00f]"));
185+
Assert.False(Validators.IsArbitraryShadow("[something-else]"));
186+
}
187+
188+
[Fact]
189+
public void IsPercentTests()
190+
{
191+
Assert.True(Validators.IsPercent("1%"));
192+
Assert.True(Validators.IsPercent("100.001%"));
193+
Assert.True(Validators.IsPercent(".01%"));
194+
Assert.True(Validators.IsPercent("0%"));
195+
196+
Assert.False(Validators.IsPercent("0"));
197+
Assert.False(Validators.IsPercent("one%"));
198+
}
199+
}

‎TailwindMerge/ClassMapFactory.cs

+138
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,138 @@
1+
using System;
2+
using System.Collections.Generic;
3+
using TailwindMerge.Models;
4+
using TailwindMerge.Interfaces;
5+
using TailwindMerge.Utilities;
6+
7+
namespace TailwindMerge
8+
{
9+
public abstract class ClassMapFactory
10+
{
11+
private const string CLASS_PART_SEPARATOR = "-";
12+
13+
public static ClassPart Create(TailwindMergeConfig config)
14+
{
15+
ClassPart classMap = new();
16+
17+
Dictionary<string, List<object>> prefixedClassGroups = GetPrefixedClassGroups(
18+
config.ClassGroups,
19+
config.Prefix
20+
);
21+
22+
foreach (var (classGroupId, classGroup) in prefixedClassGroups)
23+
{
24+
ProcessClassesRecursively(classGroup, classMap, classGroupId, config.Theme);
25+
}
26+
27+
return classMap;
28+
}
29+
30+
private static void ProcessClassesRecursively(List<object> classGroup, ClassPart classPart, string classGroupId, Dictionary<string, object> theme)
31+
{
32+
foreach (var classDefinition in classGroup)
33+
{
34+
if (classDefinition is string classDefString)
35+
{
36+
ClassPart classPartToEdit = classDefString == string.Empty ? classPart : GetPart(classPart, classDefString);
37+
classPartToEdit.SetClassGroupId(classGroupId);
38+
continue;
39+
}
40+
41+
if (classDefinition is ThemeGetter themeGetter)
42+
{
43+
Dictionary<string, dynamic> themeResult = themeGetter.Execute(theme);
44+
45+
// Convert the themeResult dictionary to a List<object>
46+
List<object> themeList = new List<object>();
47+
foreach (var kvp in themeResult)
48+
{
49+
var themeItem = new Dictionary<string, dynamic>
50+
{
51+
{ kvp.Key, kvp.Value }
52+
};
53+
themeList.Add(themeItem);
54+
}
55+
56+
ProcessClassesRecursively(
57+
themeList,
58+
classPart,
59+
classGroupId,
60+
theme
61+
);
62+
continue;
63+
}
64+
65+
if (classDefinition is IRule ruleInterface)
66+
{
67+
classPart.Validators.Add(new ClassValidator(classGroupId, ruleInterface));
68+
continue;
69+
}
70+
71+
if (classDefinition is Dictionary<string, object> classDefDict)
72+
{
73+
foreach (var (key, value) in classDefDict)
74+
{
75+
ProcessClassesRecursively(
76+
(List<object>)value,
77+
GetPart(classPart, key),
78+
classGroupId,
79+
theme
80+
);
81+
}
82+
}
83+
}
84+
}
85+
86+
private static ClassPart GetPart(ClassPart classPart, string path)
87+
{
88+
ClassPart currentClassPartObject = classPart;
89+
90+
foreach (string pathPart in path.Split(CLASS_PART_SEPARATOR))
91+
{
92+
if (!currentClassPartObject.NextPart.ContainsKey(pathPart))
93+
{
94+
currentClassPartObject.NextPart[pathPart] = new ClassPart();
95+
}
96+
97+
currentClassPartObject = currentClassPartObject.NextPart[pathPart];
98+
}
99+
100+
return currentClassPartObject;
101+
}
102+
103+
private static Dictionary<string, List<object>> GetPrefixedClassGroups(Dictionary<string, List<object>> classGroups, string prefix)
104+
{
105+
if (string.IsNullOrEmpty(prefix))
106+
{
107+
return classGroups;
108+
}
109+
110+
Dictionary<string, List<object>> output = new();
111+
112+
foreach (var (classGroupId, classGroup) in classGroups)
113+
{
114+
output[classGroupId] = classGroup.ConvertAll(classDefinition =>
115+
{
116+
if (classDefinition is string classDefString)
117+
{
118+
return prefix + classDefString;
119+
}
120+
121+
if (classDefinition is Dictionary<string, object> classDefDict)
122+
{
123+
Dictionary<string, object> prefixedClassDefinition = new();
124+
foreach (var (key, value) in classDefDict)
125+
{
126+
prefixedClassDefinition[prefix + key] = value;
127+
}
128+
return prefixedClassDefinition;
129+
}
130+
131+
return classDefinition;
132+
});
133+
}
134+
135+
return output;
136+
}
137+
}
138+
}

‎TailwindMerge/ClassValidator.cs

+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
using TailwindMerge.Interfaces; // Assuming RuleInterface is located here
2+
3+
namespace TailwindMerge
4+
{
5+
public class ClassValidator
6+
{
7+
public string ClassGroupId { get; }
8+
public IRule Rule { get; }
9+
10+
public ClassValidator(string classGroupId, IRule rule)
11+
{
12+
ClassGroupId = classGroupId;
13+
Rule = rule;
14+
}
15+
}
16+
}

‎TailwindMerge/DefaultConfig.cs

+43
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
using System;
2+
using System.Collections.Generic;
3+
using System.Text.RegularExpressions;
4+
5+
namespace TailwindMerge
6+
{
7+
/// <summary>
8+
/// Provides methods for merging Tailwind CSS classes.
9+
/// </summary>
10+
public class DefaultConfig
11+
{
12+
public DefaultConfig()
13+
{
14+
var colors = new ThemeGetter("colors");
15+
var spacing = new ThemeGetter("spacing");
16+
var blur = new ThemeGetter("blur");
17+
var brightness = new ThemeGetter("brightness");
18+
var borderColor = new ThemeGetter("borderColor");
19+
var borderRadius = new ThemeGetter("borderRadius");
20+
var borderSpacing = new ThemeGetter("borderSpacing");
21+
var borderWidth = new ThemeGetter("borderWidth");
22+
var contrast = new ThemeGetter("contrast");
23+
var grayscale = new ThemeGetter("grayscale");
24+
var hueRotate = new ThemeGetter("hueRotate");
25+
var invert = new ThemeGetter("invert");
26+
var gap = new ThemeGetter("gap");
27+
var gradientColorStops = new ThemeGetter("gradientColorStops");
28+
var gradientColorStopPositions = new ThemeGetter("gradientColorStopPositions");
29+
var inset = new ThemeGetter("inset");
30+
var margin = new ThemeGetter("margin");
31+
var opacity = new ThemeGetter("opacity");
32+
var padding = new ThemeGetter("padding");
33+
var saturate = new ThemeGetter("saturate");
34+
var scale = new ThemeGetter("scale");
35+
var sepia = new ThemeGetter("sepia");
36+
var skew = new ThemeGetter("skew");
37+
var space = new ThemeGetter("space");
38+
var translate = new ThemeGetter("translate");
39+
40+
41+
}
42+
}
43+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
using System;
2+
3+
namespace TailwindMerge.Exceptions
4+
{
5+
public class BadThemeException : Exception
6+
{
7+
public TailwindMergeConfig Config { get; }
8+
9+
public BadThemeException(string message, TailwindMergeConfig config)
10+
: base(message)
11+
{
12+
Config = config;
13+
}
14+
}
15+
}

‎TailwindMerge/Interfaces/IRule.cs

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
namespace TailwindMerge.Interfaces
2+
{
3+
public interface IRule
4+
{
5+
bool Execute(string value);
6+
}
7+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
namespace TailwindMerge.Interfaces
2+
{
3+
public interface ITailwindMergePlugin
4+
{
5+
TailwindMergeConfig Invoke(TailwindMergeConfig config);
6+
}
7+
}

‎TailwindMerge/LruCache.cs

+62
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,62 @@
1+
using System.Collections.Generic;
2+
3+
namespace TailwindMerge
4+
{
5+
public class LruCache
6+
{
7+
private Dictionary<string, string> cache = new Dictionary<string, string>();
8+
private Dictionary<string, string> previousCache = new Dictionary<string, string>();
9+
private int cacheSize = 0;
10+
11+
public LruCache(int maxCacheSize)
12+
{
13+
MaxCacheSize = maxCacheSize;
14+
}
15+
16+
public int MaxCacheSize { get; }
17+
18+
public string? Get(string key)
19+
{
20+
if (cache.ContainsKey(key))
21+
{
22+
return cache[key];
23+
}
24+
25+
if (previousCache.ContainsKey(key))
26+
{
27+
string value = previousCache[key];
28+
Update(key, value);
29+
return value;
30+
}
31+
32+
return null;
33+
}
34+
35+
public LruCache Set(string key, string value)
36+
{
37+
if (cache.ContainsKey(key))
38+
{
39+
cache[key] = value;
40+
}
41+
else
42+
{
43+
Update(key, value);
44+
}
45+
46+
return this;
47+
}
48+
49+
private void Update(string key, string value)
50+
{
51+
cache[key] = value;
52+
cacheSize++;
53+
54+
if (cacheSize > MaxCacheSize)
55+
{
56+
cacheSize = 0;
57+
previousCache = new Dictionary<string, string>(cache);
58+
cache.Clear();
59+
}
60+
}
61+
}
62+
}

‎TailwindMerge/Models/ClassContext.cs

+20
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
namespace TailwindMerge.Models
2+
{
3+
public class ClassContext
4+
{
5+
public bool IsTailwindClass { get; }
6+
public string OriginalClassName { get; }
7+
public bool HasPostfixModifier { get; }
8+
public string? ModifierId { get; }
9+
public string? ClassGroupId { get; }
10+
11+
public ClassContext(bool isTailwindClass, string originalClassName, bool hasPostfixModifier = false, string? modifierId = null, string? classGroupId = null)
12+
{
13+
IsTailwindClass = isTailwindClass;
14+
OriginalClassName = originalClassName;
15+
HasPostfixModifier = hasPostfixModifier;
16+
ModifierId = modifierId;
17+
ClassGroupId = classGroupId;
18+
}
19+
}
20+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
using System;
2+
using System.Collections.Generic;
3+
4+
namespace TailwindMerge.Models
5+
{
6+
public class ClassModifiersContext
7+
{
8+
public IReadOnlyList<string> Modifiers { get; }
9+
public bool HasImportantModifier { get; }
10+
public string BaseClassName { get; }
11+
public int? MaybePostfixModifierPosition { get; }
12+
13+
public ClassModifiersContext(IReadOnlyList<string> modifiers, bool hasImportantModifier, string baseClassName, int? maybePostfixModifierPosition)
14+
{
15+
Modifiers = modifiers ?? throw new ArgumentNullException(nameof(modifiers));
16+
HasImportantModifier = hasImportantModifier;
17+
BaseClassName = baseClassName ?? throw new ArgumentNullException(nameof(baseClassName));
18+
MaybePostfixModifierPosition = maybePostfixModifierPosition;
19+
}
20+
}
21+
}

‎TailwindMerge/Models/ClassPart.cs

+22
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
using System.Collections.Generic;
2+
3+
namespace TailwindMerge.Models
4+
{
5+
public class ClassPart
6+
{
7+
public Dictionary<string, ClassPart> NextPart { get; } = new Dictionary<string, ClassPart>();
8+
public List<ClassValidator> Validators { get; } = new List<ClassValidator>();
9+
public string? ClassGroupId { get; private set; }
10+
11+
public ClassPart(string? classGroupId = null)
12+
{
13+
ClassGroupId = classGroupId;
14+
}
15+
16+
public ClassPart SetClassGroupId(string classGroupId)
17+
{
18+
ClassGroupId = classGroupId;
19+
return this;
20+
}
21+
}
22+
}

‎TailwindMerge/Rules/AnyRule.cs

+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
using TailwindMerge.Interfaces;
2+
3+
namespace TailwindMerge.Rules
4+
{
5+
public class AnyRule : IRule
6+
{
7+
public bool Execute(string value)
8+
{
9+
return true;
10+
}
11+
}
12+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
namespace TailwindMerge.Rules
2+
{
3+
public class ArbitraryIntegerRule : ArbitraryValueRule
4+
{
5+
protected string Parameter { get; set; } = "number";
6+
7+
protected override bool TestValue(string value)
8+
{
9+
if (int.TryParse(value, out _))
10+
{
11+
return true;
12+
}
13+
return false;
14+
}
15+
}
16+
}
+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
using System.Text.RegularExpressions;
2+
3+
namespace TailwindMerge.Rules
4+
{
5+
public class ArbitraryLengthRule : ArbitraryValueRule
6+
{
7+
private const string LengthUnitRegex = @"\d+(%|px|r?em|[sdl]?v([hwib]|min|max)|pt|pc|in|cm|mm|cap|ch|ex|r?lh|cq(w|h|i|b|min|max))|\b(calc|min|max|clamp)\(.+\)|^0$";
8+
9+
protected string Parameter { get; set; } = "length";
10+
11+
protected override bool TestValue(string value)
12+
{
13+
return Regex.IsMatch(value, LengthUnitRegex);
14+
}
15+
}
16+
}
+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
namespace TailwindMerge.Rules
2+
{
3+
public class ArbitraryNumberRule : ArbitraryValueRule
4+
{
5+
protected string Parameter { get; set; } = "number";
6+
7+
protected override bool TestValue(string value)
8+
{
9+
return double.TryParse(value, out _);
10+
}
11+
}
12+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
namespace TailwindMerge.Rules
2+
{
3+
public class ArbitraryPositionRule : ArbitraryValueRule
4+
{
5+
protected string Parameter { get; set; } = "position";
6+
7+
protected override bool TestValue(string value)
8+
{
9+
return false;
10+
}
11+
}
12+
}
+14
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
using System.Text.RegularExpressions;
2+
3+
namespace TailwindMerge.Rules
4+
{
5+
public class ArbitraryShadowRule : ArbitraryValueRule
6+
{
7+
private const string ShadowRegex = @"^-?((\d+)?\.?(\d+)[a-z]+|0)_-?((\d+)?\.?(\d+)[a-z]+|0)";
8+
9+
protected override bool TestValue(string value)
10+
{
11+
return Regex.IsMatch(value, ShadowRegex);
12+
}
13+
}
14+
}
+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
namespace TailwindMerge.Rules
2+
{
3+
public class ArbitrarySizeRule : ArbitraryValueRule
4+
{
5+
protected string Parameter { get; set; } = "size";
6+
7+
protected override bool TestValue(string value)
8+
{
9+
return false;
10+
}
11+
}
12+
}
+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
namespace TailwindMerge.Rules
2+
{
3+
public class ArbitraryUrlRule : ArbitraryValueRule
4+
{
5+
protected string Parameter { get; set; } = "url";
6+
7+
protected override bool TestValue(string value)
8+
{
9+
return value.StartsWith("url(");
10+
}
11+
}
12+
}
+34
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
using System.Text.RegularExpressions;
2+
using TailwindMerge.Interfaces;
3+
4+
namespace TailwindMerge.Rules
5+
{
6+
public class ArbitraryValueRule : IRule
7+
{
8+
private const string RegexPattern = @"^\[(?:([a-z-]+):)?(.+)]$";
9+
10+
protected string Parameter { get; set; }
11+
12+
public bool Execute(string value)
13+
{
14+
var matches = Regex.Match(value, RegexPattern);
15+
16+
if (matches.Success)
17+
{
18+
if (!string.IsNullOrEmpty(matches.Groups[1].Value))
19+
{
20+
return matches.Groups[1].Value == Parameter;
21+
}
22+
23+
return TestValue(matches.Groups[2].Value);
24+
}
25+
26+
return false;
27+
}
28+
29+
protected virtual bool TestValue(string value)
30+
{
31+
return true;
32+
}
33+
}
34+
}

‎TailwindMerge/Rules/IntegerRule.cs

+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
using TailwindMerge.Interfaces;
2+
3+
namespace TailwindMerge.Rules
4+
{
5+
public class IntegerRule : IRule
6+
{
7+
public bool Execute(string value)
8+
{
9+
return int.TryParse(value, out _);
10+
}
11+
}
12+
}

‎TailwindMerge/Rules/LengthRule.cs

+18
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
using System.Text.RegularExpressions;
2+
using TailwindMerge.Interfaces;
3+
4+
namespace TailwindMerge.Rules
5+
{
6+
public class LengthRule : IRule
7+
{
8+
private const string FractionRegex = @"^\d+/\d+$";
9+
private static readonly string[] StringLengths = { "px", "full", "screen" };
10+
11+
public bool Execute(string value)
12+
{
13+
return double.TryParse(value, out _)
14+
|| StringLengths.Contains(value)
15+
|| Regex.IsMatch(value, FractionRegex);
16+
}
17+
}
18+
}

‎TailwindMerge/Rules/NeverRule.cs

+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
using TailwindMerge.Interfaces;
2+
3+
namespace TailwindMerge.Rules
4+
{
5+
public class NeverRule : IRule
6+
{
7+
public bool Execute(string value)
8+
{
9+
return false;
10+
}
11+
}
12+
}

‎TailwindMerge/Rules/NumberRule.cs

+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
using TailwindMerge.Interfaces;
2+
3+
namespace TailwindMerge.Rules
4+
{
5+
public class NumberRule : IRule
6+
{
7+
public bool Execute(string value)
8+
{
9+
return double.TryParse(value, out _);
10+
}
11+
}
12+
}

‎TailwindMerge/Rules/PercentRule.cs

+18
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
using TailwindMerge.Interfaces;
2+
3+
namespace TailwindMerge.Rules
4+
{
5+
public class PercentRule : IRule
6+
{
7+
public bool Execute(string value)
8+
{
9+
if (string.IsNullOrEmpty(value) || !value.EndsWith("%"))
10+
{
11+
return false;
12+
}
13+
14+
var numericPart = value.TrimEnd('%');
15+
return double.TryParse(numericPart, out _);
16+
}
17+
}
18+
}

‎TailwindMerge/Rules/TshirtSizeRule.cs

+15
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
using System.Text.RegularExpressions;
2+
using TailwindMerge.Interfaces;
3+
4+
namespace TailwindMerge.Rules
5+
{
6+
public class TshirtSizeRule : IRule
7+
{
8+
private const string RegexPattern = @"^(\d+(\.\d+)?)?(xs|sm|md|lg|xl)$";
9+
10+
public bool Execute(string value)
11+
{
12+
return Regex.IsMatch(value, RegexPattern);
13+
}
14+
}
15+
}

‎TailwindMerge/TailwindMerge.cs

Whitespace-only changes.

‎TailwindMerge/TailwindMerge.csproj

+2
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,8 @@
1919
<PackageLicenseFile>LICENSE.md</PackageLicenseFile>
2020

2121
<GenerateDocumentationFile>true</GenerateDocumentationFile>
22+
<!-- Suppress CS1591 warnings -->
23+
<NoWarn>$(NoWarn);CS1591</NoWarn>
2224
</PropertyGroup>
2325

2426
<ItemGroup>

‎TailwindMerge/TailwindMergeConfig.cs

Whitespace-only changes.

‎TailwindMerge/Utilities/ClassUtils.cs

+184
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,184 @@
1+
using System;
2+
using System.Collections.Generic;
3+
using TailwindMerge.Models;
4+
5+
namespace TailwindMerge.Utilities
6+
{
7+
public class ClassUtils
8+
{
9+
private const string IMPORTANT_MODIFIER = "!";
10+
private const string ARBITRARY_PROPERTY_REGEX = @"^\[(.+)]";
11+
12+
private readonly ClassPart classMap;
13+
private readonly TailwindMergeConfig _config;
14+
15+
public ClassUtils(TailwindMergeConfig config)
16+
{
17+
classMap = ClassMapFactory.Create(config);
18+
_config = config;
19+
}
20+
21+
public string? GetClassGroupId(string className)
22+
{
23+
var classParts = className.Split('-');
24+
25+
// Classes like "-inset-1" produce an empty string as the first classPart.
26+
// We assume that classes for negative values are used correctly and remove it from classParts.
27+
if (classParts[0] == "" && classParts.Length != 1)
28+
{
29+
Array.Copy(classParts, 1, classParts, 0, classParts.Length - 1);
30+
Array.Resize(ref classParts, classParts.Length - 1);
31+
}
32+
33+
return GetGroupRecursive(classParts, classMap) ?? GetGroupIdForArbitraryProperty(className);
34+
}
35+
36+
public List<string> GetConflictingClassGroupIds(string classGroupId, bool hasPostfixModifier)
37+
{
38+
var conflicts = _config.ConflictingClassGroups.GetValueOrDefault(classGroupId) ?? new List<string>();
39+
40+
if (hasPostfixModifier && _config.ConflictingClassGroupModifiers.ContainsKey(classGroupId))
41+
{
42+
conflicts.AddRange(_config.ConflictingClassGroupModifiers[classGroupId]);
43+
}
44+
45+
return conflicts;
46+
}
47+
48+
public ClassModifiersContext SplitModifiers(string className)
49+
{
50+
var separator = _config.Separator;
51+
var modifiers = new List<string>();
52+
var bracketDepth = 0;
53+
var modifierStart = 0;
54+
int? postfixModifierPosition = null;
55+
56+
for (int index = 0; index < className.Length; index++)
57+
{
58+
var currentCharacter = className[index];
59+
60+
if (bracketDepth == 0)
61+
{
62+
if (currentCharacter == separator[0] &&
63+
(separator.Length == 1 || className.Substring(index, separator.Length) == separator))
64+
{
65+
modifiers.Add(className.Substring(modifierStart, index - modifierStart));
66+
modifierStart = index + separator.Length;
67+
continue;
68+
}
69+
70+
if (currentCharacter == '/')
71+
{
72+
postfixModifierPosition = index;
73+
continue;
74+
}
75+
}
76+
77+
if (currentCharacter == '[')
78+
{
79+
bracketDepth++;
80+
}
81+
else if (currentCharacter == ']')
82+
{
83+
bracketDepth--;
84+
}
85+
}
86+
87+
var baseClassNameWithImportantModifier = modifiers.Count == 0
88+
? className
89+
: className.Substring(modifierStart);
90+
var hasImportantModifier = baseClassNameWithImportantModifier.StartsWith(IMPORTANT_MODIFIER);
91+
var baseClassName = hasImportantModifier
92+
? baseClassNameWithImportantModifier.Substring(1)
93+
: baseClassNameWithImportantModifier;
94+
int? maybePostfixModifierPosition = postfixModifierPosition > modifierStart
95+
? postfixModifierPosition - modifierStart
96+
: null;
97+
98+
return new ClassModifiersContext(modifiers, hasImportantModifier, baseClassName, maybePostfixModifierPosition);
99+
}
100+
101+
public IReadOnlyList<string> SortModifiers(IReadOnlyList<string> modifiers)
102+
{
103+
if (modifiers.Count <= 1)
104+
{
105+
return modifiers;
106+
}
107+
108+
var sortedModifiers = new List<string>();
109+
var unsortedModifiers = new List<string>();
110+
111+
foreach (var modifier in modifiers)
112+
{
113+
var isArbitraryVariant = modifier[0] == '[';
114+
if (isArbitraryVariant)
115+
{
116+
unsortedModifiers.Sort();
117+
sortedModifiers.AddRange(unsortedModifiers);
118+
sortedModifiers.Add(modifier);
119+
unsortedModifiers.Clear();
120+
}
121+
else
122+
{
123+
unsortedModifiers.Add(modifier);
124+
}
125+
}
126+
127+
unsortedModifiers.Sort();
128+
sortedModifiers.AddRange(unsortedModifiers);
129+
130+
return sortedModifiers;
131+
}
132+
133+
private string? GetGroupRecursive(string[] classParts, ClassPart classPart)
134+
{
135+
if (classParts.Length == 0)
136+
{
137+
return classPart.ClassGroupId;
138+
}
139+
140+
var currentClassPart = classParts[0];
141+
if (classPart.NextPart.TryGetValue(currentClassPart, out var nextClassPartObject))
142+
{
143+
var classGroupFromNextClassPart = GetGroupRecursive(classParts[1..], nextClassPartObject);
144+
if (classGroupFromNextClassPart != null)
145+
{
146+
return classGroupFromNextClassPart;
147+
}
148+
}
149+
150+
if (classPart.Validators.Count == 0)
151+
{
152+
return null;
153+
}
154+
155+
var classRest = string.Join("-", classParts);
156+
157+
foreach (var classValidator in classPart.Validators)
158+
{
159+
if (classValidator.Rule.Execute(classRest))
160+
{
161+
return classValidator.ClassGroupId;
162+
}
163+
}
164+
165+
return null;
166+
}
167+
168+
private string? GetGroupIdForArbitraryProperty(string className)
169+
{
170+
var match = System.Text.RegularExpressions.Regex.Match(className, ARBITRARY_PROPERTY_REGEX);
171+
if (match.Success)
172+
{
173+
var arbitraryPropertyClassName = match.Groups[1].Value;
174+
if (!string.IsNullOrEmpty(arbitraryPropertyClassName))
175+
{
176+
var property = arbitraryPropertyClassName.Substring(0, arbitraryPropertyClassName.IndexOf(':'));
177+
return "arbitrary.." + property;
178+
}
179+
}
180+
181+
return null;
182+
}
183+
}
184+
}
+17
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
namespace TailwindMerge.Utilities
2+
{
3+
public class ThemeGetter
4+
{
5+
public string Key { get; }
6+
7+
public ThemeGetter(string key)
8+
{
9+
Key = key;
10+
}
11+
12+
public Dictionary<string, dynamic> Execute(Dictionary<string, dynamic> theme)
13+
{
14+
return theme.TryGetValue(Key, out var value) ? value : new Dictionary<string, dynamic>();
15+
}
16+
}
17+
}

‎TailwindMerge/Validators.cs

+131
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,131 @@
1+
using System;
2+
using System.Collections.Generic;
3+
using System.Text.RegularExpressions;
4+
5+
namespace TailwindMerge
6+
{
7+
/// <summary>
8+
/// Provides methods for merging Tailwind CSS classes.
9+
/// </summary>
10+
public class Validators
11+
{
12+
private static readonly Regex ArbitraryValueRegex = new Regex(@"^\[(?:([a-z-]+):)?(.+)\]$", RegexOptions.IgnoreCase);
13+
private static readonly Regex FractionRegex = new Regex(@"^\d+\/\d+$");
14+
private static readonly HashSet<string> StringLengths = new HashSet<string> { "px", "full", "screen" };
15+
private static readonly Regex TshirtUnitRegex = new Regex(@"^(\d+(\.\d+)?)?(xs|sm|md|lg|xl)$");
16+
private static readonly Regex LengthUnitRegex = new Regex(
17+
@"\d+(%|px|r?em|[sdl]?v([hwib]|min|max)|pt|pc|in|cm|mm|cap|ch|ex|r?lh|cq(w|h|i|b|min|max))|\b(calc|min|max|clamp)\(.+\)|^0$");
18+
private static readonly Regex ShadowRegex = new Regex(@"^-?((\d+)?\.?(\d+)[a-z]+|0)_-?((\d+)?\.?(\d+)[a-z]+|0)");
19+
private static readonly Regex ImageRegex = new Regex(
20+
@"^(url|image|image-set|cross-fade|element|(repeating-)?(linear|radial|conic)-gradient)\(.+\)$");
21+
22+
public static bool IsLength(string value)
23+
{
24+
return IsNumber(value) || StringLengths.Contains(value) || FractionRegex.IsMatch(value);
25+
}
26+
27+
public static bool IsArbitraryLength(string value)
28+
{
29+
return GetIsArbitraryValue(value, "length", IsLengthOnly);
30+
}
31+
32+
public static bool IsNumber(string value)
33+
{
34+
return !string.IsNullOrEmpty(value) && double.TryParse(value, out _);
35+
}
36+
37+
public static bool IsArbitraryNumber(string value)
38+
{
39+
return GetIsArbitraryValue(value, "number", IsNumber);
40+
}
41+
42+
public static bool IsInteger(string value)
43+
{
44+
return !string.IsNullOrEmpty(value) && int.TryParse(value, out _);
45+
}
46+
47+
public static bool IsPercent(string value)
48+
{
49+
return value.EndsWith("%") && IsNumber(value[..^1]);
50+
}
51+
52+
public static bool IsArbitraryValue(string value)
53+
{
54+
return ArbitraryValueRegex.IsMatch(value);
55+
}
56+
57+
public static bool IsTshirtSize(string value)
58+
{
59+
return TshirtUnitRegex.IsMatch(value);
60+
}
61+
62+
private static readonly HashSet<string> SizeLabels = new HashSet<string> { "length", "size", "percentage" };
63+
64+
public static bool IsArbitrarySize(string value)
65+
{
66+
return GetIsArbitraryValue(value, SizeLabels, IsNever);
67+
}
68+
69+
public static bool IsArbitraryPosition(string value)
70+
{
71+
return GetIsArbitraryValue(value, "position", IsNever);
72+
}
73+
74+
private static readonly HashSet<string> ImageLabels = new HashSet<string> { "image", "url" };
75+
76+
public static bool IsArbitraryImage(string value)
77+
{
78+
return GetIsArbitraryValue(value, ImageLabels, IsImage);
79+
}
80+
81+
public static bool IsArbitraryShadow(string value)
82+
{
83+
return GetIsArbitraryValue(value, string.Empty, IsShadow);
84+
}
85+
86+
public static bool IsAny()
87+
{
88+
return true;
89+
}
90+
91+
private static bool GetIsArbitraryValue(string value, object label, Func<string, bool> testValue)
92+
{
93+
var match = ArbitraryValueRegex.Match(value);
94+
95+
if (match.Success)
96+
{
97+
if (!string.IsNullOrEmpty(match.Groups[1].Value))
98+
{
99+
if (label is string strLabel)
100+
return match.Groups[1].Value == strLabel;
101+
if (label is HashSet<string> setLabel)
102+
return setLabel.Contains(match.Groups[1].Value);
103+
}
104+
105+
return testValue(match.Groups[2].Value);
106+
}
107+
108+
return false;
109+
}
110+
111+
private static bool IsLengthOnly(string value)
112+
{
113+
return LengthUnitRegex.IsMatch(value);
114+
}
115+
116+
private static bool IsNever(string value)
117+
{
118+
return false;
119+
}
120+
121+
private static bool IsShadow(string value)
122+
{
123+
return ShadowRegex.IsMatch(value);
124+
}
125+
126+
private static bool IsImage(string value)
127+
{
128+
return ImageRegex.IsMatch(value);
129+
}
130+
}
131+
}

0 commit comments

Comments
 (0)
Please sign in to comment.