Skip to content

Commit c9f7a2f

Browse files
authored
Merge pull request #23 from javierav/add-merge
2 parents d5bc1a9 + f34a069 commit c9f7a2f

File tree

3 files changed

+97
-7
lines changed

3 files changed

+97
-7
lines changed

CHANGELOG.md

+3
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,6 @@
1+
## Unreleased
2+
- Add support for merging ([#23](https://github.com/avo-hq/class_variants/pull/23))
3+
14
## 1.0.0 (2024-11-13)
25
- Add support for slots ([#15](https://github.com/avo-hq/class_variants/pull/15))
36
- Allow passing additional classes when render ([#17](https://github.com/avo-hq/class_variants/pull/17))

lib/class_variants/instance.rb

+23-7
Original file line numberDiff line numberDiff line change
@@ -1,20 +1,36 @@
11
module ClassVariants
22
class Instance
3-
def initialize(**options, &block)
3+
def initialize(...)
4+
@bases = []
5+
@variants = []
6+
@defaults = {}
7+
8+
merge(...)
9+
end
10+
11+
def merge(**options, &block)
412
raise ArgumentError, "Use of hash config and code block is not supported" if !options.empty? && block_given?
513

6-
@base = options.empty? ? {} : {default: options.fetch(:base, nil)}
7-
@variants = expand_variants(options.fetch(:variants, {})) + expand_compound_variants(options.fetch(:compound_variants, []))
8-
@defaults = options.fetch(:defaults, {})
14+
(base = options.fetch(:base, nil)) && @bases << {class: base, slot: :default}
15+
@variants += [
16+
expand_variants(options.fetch(:variants, {})),
17+
expand_compound_variants(options.fetch(:compound_variants, []))
18+
].inject(:+)
19+
@defaults.merge!(options.fetch(:defaults, {}))
920

1021
instance_eval(&block) if block_given?
22+
23+
self
1124
end
1225

1326
def render(slot = :default, **overrides)
1427
classes = overrides.delete(:class)
28+
result = []
1529

1630
# Start with our default classes
17-
result = [@base[slot]]
31+
@bases.each do |base|
32+
result << base[:class] if base[:slot] == slot
33+
end
1834

1935
# Then merge the passed in overrides on top of the defaults
2036
criteria = @defaults.merge(overrides)
@@ -44,10 +60,10 @@ def base(klass = nil, &block)
4460

4561
if block_given?
4662
with_slots(&block).each do |slot|
47-
@base[slot[:slot]] = slot[:class]
63+
@bases << slot
4864
end
4965
else
50-
@base[:default] = klass
66+
@bases << {slot: :default, class: klass}
5167
end
5268
end
5369

test/merge_test.rb

+71
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,71 @@
1+
require "test_helper"
2+
3+
class MergeTest < Minitest::Test
4+
def test_hash_merge
5+
cv = ClassVariants.build(
6+
base: "rounded",
7+
variants: {
8+
color: {
9+
primary: "bg-blue-500",
10+
secondary: "bg-purple-500",
11+
success: "bg-green-500"
12+
}
13+
},
14+
default: {
15+
color: :primary
16+
}
17+
)
18+
cv.merge(
19+
base: "border",
20+
variants: {
21+
color: {
22+
primary: "bg-blue-700",
23+
secondary: "bg-purple-700"
24+
}
25+
},
26+
defaults: {
27+
color: :secondary
28+
}
29+
)
30+
31+
assert_equal "rounded border bg-purple-500 bg-purple-700", cv.render
32+
end
33+
34+
def test_block_merge
35+
cv = ClassVariants.build do
36+
base do
37+
slot :root, class: "rounded"
38+
slot :title, class: "font-bold"
39+
end
40+
41+
variant variant: :outlined do
42+
slot :root, class: "border-red-700"
43+
slot :title, class: "text-red-700"
44+
end
45+
46+
variant variant: :filled do
47+
slot :root, class: "bg-red-100"
48+
slot :title, class: "text-red-900"
49+
end
50+
51+
defaults variant: :outlined
52+
end
53+
54+
cv.merge do
55+
base do
56+
slot :root, class: "mb-4"
57+
slot :title, class: "mb-1"
58+
end
59+
60+
variant variant: :filled do
61+
slot :root, class: "dark:bg-red-800"
62+
slot :title, class: "dark:text-red-50"
63+
end
64+
65+
defaults variant: :filled
66+
end
67+
68+
assert_equal "rounded mb-4 bg-red-100 dark:bg-red-800", cv.render(:root)
69+
assert_equal "font-bold mb-1 text-red-900 dark:text-red-50", cv.render(:title)
70+
end
71+
end

0 commit comments

Comments
 (0)