1
0
Fork 0
mirror of https://gitlab.com/bramw/baserow.git synced 2025-04-04 21:25:24 +00:00

Configure Link decoration in Builder Theme

This commit is contained in:
Tsering Paljor 2025-03-18 18:41:04 +04:00 committed by Jérémie Pardou
parent 7cec473987
commit 722bd54323
22 changed files with 760 additions and 53 deletions

View file

@ -37,16 +37,37 @@ class DynamicConfigBlockSerializer(serializers.Serializer):
if not isinstance(theme_config_block_type_name, list):
theme_config_block_type_name = [theme_config_block_type_name]
for prop, type_name in zip(property_name, theme_config_block_type_name):
theme_config_block_type = theme_config_block_registry.get(type_name)
self.fields[prop] = theme_config_block_type.get_serializer_class(
request_serializer=request_serializer
)(**({"help_text": f"Styles overrides for {prop}"} | serializer_kwargs))
for prop, type_names in zip(property_name, theme_config_block_type_name):
if not isinstance(type_names, list):
type_names = [type_names]
config_blocks = (
theme_config_block_registry.get(type_name) for type_name in type_names
)
serializer_class = combine_theme_config_blocks_serializer_class(
config_blocks,
request_serializer=request_serializer,
name="SubConfigBlockSerializer",
)
self.fields[prop] = serializer_class(**serializer_kwargs)
all_type_names = "".join(
[
"And".join(sub.capitalize() for sub in p)
if isinstance(p, list)
else p.capitalize()
for p in theme_config_block_type_name
]
)
# Dynamically create the Meta class with ref name to prevent collision
class DynamicMeta:
type_names = "".join([p.capitalize() for p in theme_config_block_type_name])
ref_name = f"{type_names}ConfigBlockSerializer"
type_names = all_type_names
ref_name = f"{all_type_names}ConfigBlockSerializer"
meta_ref_name = f"{all_type_names}ConfigBlockSerializer"
print("Serializer for ", f"{all_type_names}ConfigBlockSerializer")
self.Meta = DynamicMeta
@ -72,6 +93,41 @@ def serialize_builder_theme(builder: Builder) -> dict:
return theme
def combine_theme_config_blocks_serializer_class(
theme_config_blocks,
request_serializer=False,
name="CombinedThemeConfigBlocksSerializer",
) -> serializers.Serializer:
"""
This helper function generates one single serializer that contains all the fields
of all the theme config blocks. The API always communicates all theme properties
flat in one single object.
:return: The generated serializer.
"""
attrs = {}
for theme_config_block in theme_config_blocks:
serializer = theme_config_block.get_serializer_class(
request_serializer=request_serializer
)
serializer_fields = serializer().get_fields()
for name, field in serializer_fields.items():
attrs[name] = field
class Meta:
ref_name = "".join(t.type.capitalize() for t in theme_config_blocks) + name
meta_ref_name = "".join(t.type.capitalize() for t in theme_config_blocks) + name
attrs["Meta"] = Meta
class_object = type(name, (serializers.Serializer,), attrs)
return class_object
@cache
def get_combined_theme_config_blocks_serializer_class(
request_serializer=False,
@ -90,28 +146,10 @@ def get_combined_theme_config_blocks_serializer_class(
"imported before the theme config blocks have been registered."
)
attrs = {}
for theme_config_block in theme_config_block_registry.get_all():
serializer = theme_config_block.get_serializer_class(
request_serializer=request_serializer
)
serializer_fields = serializer().get_fields()
for name, field in serializer_fields.items():
attrs[name] = field
class Meta:
meta_ref_name = "combined_theme_config_blocks_serializer"
attrs["Meta"] = Meta
class_object = type(
"CombinedThemeConfigBlocksSerializer", (serializers.Serializer,), attrs
return combine_theme_config_blocks_serializer_class(
theme_config_block_registry.get_all(), request_serializer=request_serializer
)
return class_object
CombinedThemeConfigBlocksSerializer = (
get_combined_theme_config_blocks_serializer_class()

View file

@ -257,7 +257,6 @@ class BuilderConfig(AppConfig):
ImageThemeConfigBlockType,
InputThemeConfigBlockType,
LinkThemeConfigBlockType,
MenuThemeConfigBlockType,
PageThemeConfigBlockType,
TableThemeConfigBlockType,
TypographyThemeConfigBlockType,
@ -266,12 +265,11 @@ class BuilderConfig(AppConfig):
theme_config_block_registry.register(ColorThemeConfigBlockType())
theme_config_block_registry.register(TypographyThemeConfigBlockType())
theme_config_block_registry.register(ButtonThemeConfigBlockType())
theme_config_block_registry.register(LinkThemeConfigBlockType())
theme_config_block_registry.register(ImageThemeConfigBlockType())
theme_config_block_registry.register(PageThemeConfigBlockType())
theme_config_block_registry.register(InputThemeConfigBlockType())
theme_config_block_registry.register(TableThemeConfigBlockType())
theme_config_block_registry.register(MenuThemeConfigBlockType())
theme_config_block_registry.register(LinkThemeConfigBlockType())
from .workflow_actions.registries import builder_workflow_action_type_registry
from .workflow_actions.workflow_action_types import (

View file

@ -2011,7 +2011,8 @@ class MenuElementType(ElementType):
DynamicConfigBlockSerializer,
)
from baserow.contrib.builder.theme.theme_config_block_types import (
MenuThemeConfigBlockType,
ButtonThemeConfigBlockType,
LinkThemeConfigBlockType,
)
overrides = {
@ -2019,7 +2020,9 @@ class MenuElementType(ElementType):
"styles": DynamicConfigBlockSerializer(
required=False,
property_name="menu",
theme_config_block_type_name=MenuThemeConfigBlockType.type,
theme_config_block_type_name=[
[ButtonThemeConfigBlockType.type, LinkThemeConfigBlockType.type]
],
serializer_kwargs={"required": False},
),
}

View file

@ -0,0 +1,116 @@
# Generated by Django 5.0.9 on 2025-03-18 13:03
from django.db import migrations
import baserow.core.fields
class Migration(migrations.Migration):
dependencies = [
("builder", "0054_simplecontainerelement"),
]
operations = [
migrations.AddField(
model_name="linkthemeconfigblock",
name="link_active_text_decoration",
field=baserow.core.fields.MultipleFlagField(
db_default="1000",
default="1000",
help_text="The text decoration flags [underline, strike, uppercase, italic]",
max_length=4,
num_flags=4,
),
),
migrations.AddField(
model_name="linkthemeconfigblock",
name="link_default_text_decoration",
field=baserow.core.fields.MultipleFlagField(
db_default="1000",
default="1000",
help_text="The text decoration flags [underline, strike, uppercase, italic]",
max_length=4,
num_flags=4,
),
),
migrations.AddField(
model_name="linkthemeconfigblock",
name="link_hover_text_decoration",
field=baserow.core.fields.MultipleFlagField(
db_default="1000",
default="1000",
help_text="The text decoration flags [underline, strike, uppercase, italic]",
max_length=4,
num_flags=4,
),
),
migrations.AddField(
model_name="typographythemeconfigblock",
name="heading_1_text_decoration",
field=baserow.core.fields.MultipleFlagField(
db_default="0000",
default="0000",
help_text="The text decoration flags [underline, strike, uppercase, italic]",
max_length=4,
num_flags=4,
),
),
migrations.AddField(
model_name="typographythemeconfigblock",
name="heading_2_text_decoration",
field=baserow.core.fields.MultipleFlagField(
db_default="0000",
default="0000",
help_text="The text decoration flags [underline, strike, uppercase, italic]",
max_length=4,
num_flags=4,
),
),
migrations.AddField(
model_name="typographythemeconfigblock",
name="heading_3_text_decoration",
field=baserow.core.fields.MultipleFlagField(
db_default="0000",
default="0000",
help_text="The text decoration flags [underline, strike, uppercase, italic]",
max_length=4,
num_flags=4,
),
),
migrations.AddField(
model_name="typographythemeconfigblock",
name="heading_4_text_decoration",
field=baserow.core.fields.MultipleFlagField(
db_default="0000",
default="0000",
help_text="The text decoration flags [underline, strike, uppercase, italic]",
max_length=4,
num_flags=4,
),
),
migrations.AddField(
model_name="typographythemeconfigblock",
name="heading_5_text_decoration",
field=baserow.core.fields.MultipleFlagField(
db_default="0000",
default="0000",
help_text="The text decoration flags [underline, strike, uppercase, italic]",
max_length=4,
num_flags=4,
),
),
migrations.AddField(
model_name="typographythemeconfigblock",
name="heading_6_text_decoration",
field=baserow.core.fields.MultipleFlagField(
db_default="0000",
default="0000",
help_text="The text decoration flags [underline, strike, uppercase, italic]",
max_length=4,
num_flags=4,
),
),
migrations.DeleteModel(
name="MenuThemeConfigBlock",
),
]

View file

@ -8,7 +8,7 @@ from baserow.contrib.builder.constants import (
FontWeights,
HorizontalAlignments,
)
from baserow.core.fields import AutoOneToOneField
from baserow.core.fields import AutoOneToOneField, MultipleFlagField
from baserow.core.user_files.models import UserFile
@ -84,6 +84,12 @@ class TypographyThemeConfigBlock(ThemeConfigBlock):
max_length=10,
default=HorizontalAlignments.LEFT,
)
heading_1_text_decoration = MultipleFlagField(
default=[False, False, False, False],
num_flags=4,
db_default="0000",
help_text=("The text decoration flags [underline, strike, uppercase, italic]"),
)
heading_2_font_family = models.CharField(
max_length=250,
default="inter",
@ -103,6 +109,12 @@ class TypographyThemeConfigBlock(ThemeConfigBlock):
max_length=10,
default=HorizontalAlignments.LEFT,
)
heading_2_text_decoration = MultipleFlagField(
default=[False, False, False, False],
num_flags=4,
db_default="0000",
help_text=("The text decoration flags [underline, strike, uppercase, italic]"),
)
heading_3_font_family = models.CharField(
max_length=250,
default="inter",
@ -122,6 +134,12 @@ class TypographyThemeConfigBlock(ThemeConfigBlock):
max_length=10,
default=HorizontalAlignments.LEFT,
)
heading_3_text_decoration = MultipleFlagField(
default=[False, False, False, False],
num_flags=4,
db_default="0000",
help_text=("The text decoration flags [underline, strike, uppercase, italic]"),
)
heading_4_font_family = models.CharField(
max_length=250,
default="inter",
@ -141,6 +159,12 @@ class TypographyThemeConfigBlock(ThemeConfigBlock):
max_length=10,
default=HorizontalAlignments.LEFT,
)
heading_4_text_decoration = MultipleFlagField(
default=[False, False, False, False],
num_flags=4,
db_default="0000",
help_text=("The text decoration flags [underline, strike, uppercase, italic]"),
)
heading_5_font_family = models.CharField(
max_length=250,
default="inter",
@ -160,6 +184,12 @@ class TypographyThemeConfigBlock(ThemeConfigBlock):
max_length=10,
default=HorizontalAlignments.LEFT,
)
heading_5_text_decoration = MultipleFlagField(
default=[False, False, False, False],
num_flags=4,
db_default="0000",
help_text=("The text decoration flags [underline, strike, uppercase, italic]"),
)
heading_6_font_family = models.CharField(
max_length=250,
default="inter",
@ -179,6 +209,12 @@ class TypographyThemeConfigBlock(ThemeConfigBlock):
max_length=10,
default=HorizontalAlignments.LEFT,
)
heading_6_text_decoration = MultipleFlagField(
default=[False, False, False, False],
num_flags=4,
db_default="0000",
help_text=("The text decoration flags [underline, strike, uppercase, italic]"),
)
class ButtonThemeConfigBlockMixin(models.Model):
@ -318,6 +354,24 @@ class LinkThemeConfigBlockMixin(models.Model):
blank=True,
help_text="The hover color of links when active",
)
link_default_text_decoration = MultipleFlagField(
default=[True, False, False, False],
num_flags=4,
db_default="1000",
help_text=("The text decoration flags [underline, strike, uppercase, italic]"),
)
link_hover_text_decoration = MultipleFlagField(
default=[True, False, False, False],
num_flags=4,
db_default="1000",
help_text=("The text decoration flags [underline, strike, uppercase, italic]"),
)
link_active_text_decoration = MultipleFlagField(
default=[True, False, False, False],
num_flags=4,
db_default="1000",
help_text=("The text decoration flags [underline, strike, uppercase, italic]"),
)
class Meta:
abstract = True
@ -573,9 +627,3 @@ class TableThemeConfigBlock(ThemeConfigBlock):
table_horizontal_separator_size = models.SmallIntegerField(
default=1, help_text="Table horizontal separator size"
)
class MenuThemeConfigBlock(
LinkThemeConfigBlockMixin, ButtonThemeConfigBlockMixin, ThemeConfigBlock
):
pass

View file

@ -15,7 +15,6 @@ from .models import (
ImageThemeConfigBlock,
InputThemeConfigBlock,
LinkThemeConfigBlock,
MenuThemeConfigBlock,
PageThemeConfigBlock,
TableThemeConfigBlock,
ThemeConfigBlock,
@ -33,6 +32,41 @@ class TypographyThemeConfigBlockType(ThemeConfigBlockType):
type = "typography"
model_class = TypographyThemeConfigBlock
@property
def serializer_field_overrides(self):
return {
"heading_1_text_decoration": serializers.ListField(
child=serializers.BooleanField(),
help_text="Text decoration: [underline, stroke, uppercase, italic]",
required=False,
),
"heading_2_text_decoration": serializers.ListField(
child=serializers.BooleanField(),
help_text="Text decoration: [underline, stroke, uppercase, italic]",
required=False,
),
"heading_3_text_decoration": serializers.ListField(
child=serializers.BooleanField(),
help_text="Text decoration: [underline, stroke, uppercase, italic]",
required=False,
),
"heading_4_text_decoration": serializers.ListField(
child=serializers.BooleanField(),
help_text="Text decoration: [underline, stroke, uppercase, italic]",
required=False,
),
"heading_5_text_decoration": serializers.ListField(
child=serializers.BooleanField(),
help_text="Text decoration: [underline, stroke, uppercase, italic]",
required=False,
),
"heading_6_text_decoration": serializers.ListField(
child=serializers.BooleanField(),
help_text="Text decoration: [underline, stroke, uppercase, italic]",
required=False,
),
}
def import_serialized(
self,
parent: Any,
@ -64,6 +98,26 @@ class LinkThemeConfigBlockType(ThemeConfigBlockType):
type = "link"
model_class = LinkThemeConfigBlock
@property
def serializer_field_overrides(self):
return {
"link_default_text_decoration": serializers.ListField(
child=serializers.BooleanField(),
help_text="Default text decoration: [underline, stroke, uppercase, italic]",
required=False,
),
"link_hover_text_decoration": serializers.ListField(
child=serializers.BooleanField(),
help_text="Hover text decoration: [underline, stroke, uppercase, italic]",
required=False,
),
"link_active_text_decoration": serializers.ListField(
child=serializers.BooleanField(),
help_text="Active text decoration: [underline, stroke, uppercase, italic]",
required=False,
),
}
class ImageThemeConfigBlockType(ThemeConfigBlockType):
type = "image"
@ -183,8 +237,3 @@ class InputThemeConfigBlockType(ThemeConfigBlockType):
class TableThemeConfigBlockType(ThemeConfigBlockType):
type = "table"
model_class = TableThemeConfigBlock
class MenuThemeConfigBlockType(ThemeConfigBlockType):
type = "menu"
model_class = MenuThemeConfigBlock

View file

@ -257,3 +257,84 @@ class LenientDecimalField(models.Field):
**kwargs,
}
)
def default_boolean_list(num_flags):
"""Returns a default list of False values"""
return [False] * num_flags
class MultipleFlagField(models.CharField):
"""Stores a list of booleans as a binary string"""
def __init__(self, num_flags=8, default=None, *args, **kwargs):
self.num_flags = num_flags
kwargs.setdefault("max_length", num_flags) # Ensures max length is set
# Handle list-based default values properly
if default is None:
default = default_boolean_list(num_flags)
if isinstance(default, list):
if len(default) != num_flags:
raise ValueError(f"Default list must have exactly {num_flags} elements")
# Convert list to string representation
kwargs["default"] = "".join("1" if flag else "0" for flag in default)
elif isinstance(default, str):
if len(default) != num_flags or not set(default).issubset({"0", "1"}):
raise ValueError(
f"Default string must be exactly {num_flags} characters of "
"'0' or '1'"
)
kwargs["default"] = default
else:
raise ValueError(
"Default must be a list of booleans, a binary string, or None"
)
super().__init__(*args, **kwargs)
def from_db_value(self, value, expression, connection):
"""
Converts the stored binary string into a list of booleans when retrieving
from the database
"""
if value is None:
return default_boolean_list(self.num_flags)
return [char == "1" for char in value]
def to_python(self, value):
"""Ensures the value is always returned as a list of booleans"""
if isinstance(value, list):
return value
return [char == "1" for char in value]
def get_prep_value(self, value):
"""Converts the list of booleans into a binary string for database storage"""
if isinstance(value, str):
# If Django passes the default value as a string, assume it's already in
# correct format
if len(value) != self.num_flags or not set(value).issubset({"0", "1"}):
raise ValueError(
f"Stored string must have exactly {self.num_flags} characters of "
"'0' or '1'"
)
return value
elif isinstance(value, list):
if len(value) != self.num_flags:
raise ValueError(f"List must have exactly {self.num_flags} elements")
return "".join("1" if flag else "0" for flag in value)
else:
raise ValueError(
"Value must be a list of booleans or a valid binary string"
)
def deconstruct(self):
"""Ensures Django migrations correctly store and restore num_flags"""
name, path, args, kwargs = super().deconstruct()
kwargs["num_flags"] = self.num_flags # Add num_flags explicitly
return name, path, args, kwargs

View file

@ -621,31 +621,37 @@ def test_builder_application_export(data_fixture):
"heading_1_font_weight": "bold",
"heading_1_text_color": "#070810ff",
"heading_1_text_alignment": "left",
"heading_1_text_decoration": [False, False, False, False],
"heading_2_font_family": "inter",
"heading_2_font_size": 20,
"heading_2_font_weight": "semi-bold",
"heading_2_text_color": "#070810ff",
"heading_2_text_alignment": "left",
"heading_2_text_decoration": [False, False, False, False],
"heading_3_font_family": "inter",
"heading_3_font_size": 16,
"heading_3_font_weight": "medium",
"heading_3_text_color": "#070810ff",
"heading_3_text_alignment": "left",
"heading_3_text_decoration": [False, False, False, False],
"heading_4_font_family": "inter",
"heading_4_font_size": 16,
"heading_4_font_weight": "medium",
"heading_4_text_color": "#070810ff",
"heading_4_text_alignment": "left",
"heading_4_text_decoration": [False, False, False, False],
"heading_5_font_family": "inter",
"heading_5_font_size": 14,
"heading_5_font_weight": "regular",
"heading_5_text_color": "#070810ff",
"heading_5_text_alignment": "left",
"heading_5_text_decoration": [False, False, False, False],
"heading_6_font_family": "inter",
"heading_6_font_size": 14,
"heading_6_font_weight": "regular",
"heading_6_text_color": "#202128",
"heading_6_text_alignment": "left",
"heading_6_text_decoration": [False, False, False, False],
"button_font_family": "inter",
"button_font_size": 13,
"button_font_weight": "regular",
@ -673,6 +679,9 @@ def test_builder_application_export(data_fixture):
"link_text_color": "primary",
"link_hover_text_color": "#96baf6ff",
"link_active_text_color": "#275d9f",
"link_active_text_decoration": [True, False, False, False],
"link_default_text_decoration": [True, False, False, False],
"link_hover_text_decoration": [True, False, False, False],
"image_alignment": "left",
"image_border_radius": 0,
"image_max_width": 100,

View file

@ -0,0 +1,8 @@
{
"type": "feature",
"message": "Added theme setting to configure Link decoration.",
"domain": "builder",
"issue_number": 3515,
"bullet_points": [],
"created_at": "2025-03-14"
}

View file

@ -0,0 +1,49 @@
<template>
<div class="text-decoration-selector">
<SwitchButton
:value="value[0]"
:icon="'iconoir-underline'"
:title="$t('textDecorationSelector.underline')"
@input="toggle(0)"
/>
<SwitchButton
:value="value[1]"
:icon="'iconoir-strikethrough'"
:title="$t('textDecorationSelector.stroke')"
@input="toggle(1)"
/>
<SwitchButton
:value="value[2]"
:icon="'iconoir-text'"
:title="$t('textDecorationSelector.uppercase')"
@input="toggle(2)"
/>
<SwitchButton
:value="value[3]"
:icon="'iconoir-italic'"
:title="$t('textDecorationSelector.italic')"
@input="toggle(3)"
/>
</div>
</template>
<script>
export default {
name: 'TextDecorationSelector',
props: {
value: {
type: Array,
required: false,
default: () => [false, false, false, false],
},
},
methods: {
toggle(index) {
this.$emit(
'input',
this.value.map((v, i) => (i === index ? !v : v))
)
},
},
}
</script>

View file

@ -99,6 +99,22 @@
/>
</template>
</FormGroup>
<FormGroup
horizontal-narrow
small-label
class="margin-bottom-2"
:label="$t('linkThemeConfigBlock.decoration')"
>
<TextDecorationSelector
v-model="values.link_default_text_decoration"
/>
<template #after-input>
<ResetButton
v-model="values.link_default_text_decoration"
:default-value="theme?.link_default_text_decoration"
/>
</template>
</FormGroup>
</template>
<template #preview>
<ABLink url="">{{ $t('linkThemeConfigBlock.link') }}</ABLink>
@ -126,6 +142,20 @@
/>
</template>
</FormGroup>
<FormGroup
horizontal-narrow
small-label
class="margin-bottom-2"
:label="$t('linkThemeConfigBlock.decoration')"
>
<TextDecorationSelector v-model="values.link_hover_text_decoration" />
<template #after-input>
<ResetButton
v-model="values.link_hover_text_decoration"
:default-value="theme?.link_hover_text_decoration"
/>
</template>
</FormGroup>
</template>
<template #preview>
<ABLink url="" class="ab-link--force-hover">
@ -155,6 +185,22 @@
/>
</template>
</FormGroup>
<FormGroup
horizontal-narrow
small-label
class="margin-bottom-2"
:label="$t('linkThemeConfigBlock.decoration')"
>
<TextDecorationSelector
v-model="values.link_active_text_decoration"
/>
<template #after-input>
<ResetButton
v-model="values.link_active_text_decoration"
:default-value="theme?.link_active_text_decoration"
/>
</template>
</FormGroup>
</template>
<template #preview>
<ABLink url="" class="ab-link--force-active">
@ -174,6 +220,7 @@ import HorizontalAlignmentsSelector from '@baserow/modules/builder/components/Ho
import FontFamilySelector from '@baserow/modules/builder/components/FontFamilySelector'
import FontWeightSelector from '@baserow/modules/builder/components/FontWeightSelector'
import PixelValueSelector from '@baserow/modules/builder/components/PixelValueSelector'
import TextDecorationSelector from '@baserow/modules/builder/components/TextDecorationSelector'
import {
required,
integer,
@ -199,6 +246,7 @@ export default {
FontFamilySelector,
FontWeightSelector,
PixelValueSelector,
TextDecorationSelector,
},
mixins: [themeConfigBlock],
setup() {
@ -226,6 +274,9 @@ export default {
link_font_family: this.theme?.link_font_family,
link_font_weight: this.theme?.link_font_weight,
link_font_size: this.theme?.link_font_size,
link_default_text_decoration: this.theme?.link_default_text_decoration,
link_hover_text_decoration: this.theme?.link_hover_text_decoration,
link_active_text_decoration: this.theme?.link_active_text_decoration,
},
}
},
@ -267,6 +318,9 @@ export default {
link_active_text_color: {},
link_font_family: {},
link_font_weight: {},
link_default_text_decoration: {},
link_hover_text_decoration: {},
link_active_text_decoration: {},
},
}
},

View file

@ -204,6 +204,20 @@
/>
</template>
</FormGroup>
<FormGroup
horizontal-narrow
small-label
class="margin-bottom-2"
:label="$t('typographyThemeConfigBlock.decoration')"
>
<TextDecorationSelector
v-model="values[`heading_${level}_text_decoration`]" />
<template #after-input>
<ResetButton
v-model="values[`heading_${level}_text_decoration`]"
:default-value="theme?.[`heading_${level}_text_decoration`]"
/> </template
></FormGroup>
</template>
<template #preview>
<ABHeading
@ -234,6 +248,7 @@ import HorizontalAlignmentsSelector from '@baserow/modules/builder/components/Ho
import FontFamilySelector from '@baserow/modules/builder/components/FontFamilySelector'
import FontWeightSelector from '@baserow/modules/builder/components/FontWeightSelector'
import PixelValueSelector from '@baserow/modules/builder/components/PixelValueSelector'
import TextDecorationSelector from '@baserow/modules/builder/components/TextDecorationSelector'
import { DEFAULT_FONT_SIZE_PX } from '@baserow/modules/builder/defaultStyles'
const fontSizeMin = 1
@ -250,6 +265,7 @@ export default {
FontFamilySelector,
FontWeightSelector,
PixelValueSelector,
TextDecorationSelector,
},
mixins: [themeConfigBlock],
setup() {
@ -269,6 +285,7 @@ export default {
o[`heading_${i}_font_family`] = ''
o[`heading_${i}_font_weight`] = ''
o[`heading_${i}_text_alignment`] = ''
o[`heading_${i}_text_decoration`] = [false, false, false]
return o
}, {}),
},

View file

@ -140,11 +140,11 @@
"elementInProgress": "Adding element..."
},
"addElementCategory": {
"suggestedElement": "Suggested elements",
"baseElement": "Base elements",
"layoutElement": "Layout elements",
"formElement": "Form elements"
},
"suggestedElement": "Suggested elements",
"baseElement": "Base elements",
"layoutElement": "Layout elements",
"formElement": "Form elements"
},
"elementMenu": {
"moveUp": "Move up",
"moveDown": "Move down",
@ -563,6 +563,7 @@
"weight": "Weight",
"textAlignment": "Alignment",
"bodyLabel": "Body",
"decoration": "Text decoration",
"fontFamily": "Font"
},
"fontWeightType": {
@ -595,6 +596,10 @@
"size": "Font size",
"weight": "Font weight"
},
"linkDecorations": {
"normal": "Normal",
"plain": "Plain"
},
"linkThemeConfigBlock": {
"color": "Color",
"link": "Link",
@ -604,7 +609,8 @@
"alignment": "Alignment",
"fontFamily": "Font",
"size": "Font size",
"weight": "Font weight"
"weight": "Font weight",
"decoration": "Text decoration"
},
"inputThemeConfigBlock": {
"label": "Label",
@ -972,5 +978,11 @@
"authProviderWithModal": {
"authProviderInError": "Please edit this provider to fix the error.",
"title": "Edit provider: {name}"
},
"textDecorationSelector": {
"underline": "Underline",
"stroke": "Stroke",
"italic": "Italic",
"uppercase": "Uppercase"
}
}

View file

@ -289,6 +289,36 @@ export class TypographyThemeConfigBlockType extends ThemeConfigBlockType {
`heading_${level}_font_weight`,
`--heading-h${level}-font-weight`
)
style.addIfExists(
theme,
`heading_${level}_text_decoration`,
`--heading-h${level}-text-decoration`,
(v) => {
const value = []
if (v[0]) {
value.push('underline')
}
if (v[1]) {
value.push('line-through')
}
if (value.length === 0) {
return 'none'
}
return value.join(' ')
}
)
style.addIfExists(
theme,
`heading_${level}_text_decoration`,
`--heading-h${level}-text-transform`,
(v) => (v[2] ? 'uppercase' : 'none')
)
style.addIfExists(
theme,
`heading_${level}_text_decoration`,
`--heading-h${level}-font-style`,
(v) => (v[3] ? 'italic' : 'none')
)
})
style.addPixelValueIfExists(theme, `body_font_size`)
style.addColorIfExists(theme, `body_text_color`)
@ -398,6 +428,96 @@ export class LinkThemeConfigBlockType extends ThemeConfigBlockType {
const fontFamilyType = this.app.$registry.get('fontFamily', v)
return `"${fontFamilyType.name}","${fontFamilyType.safeFont}"`
})
style.addIfExists(
theme,
'link_default_text_decoration',
'--link-default-text-decoration',
(v) => {
const value = []
if (v[0]) {
value.push('underline')
}
if (v[1]) {
value.push('line-through')
}
if (value.length === 0) {
return 'none'
}
return value.join(' ')
}
)
style.addIfExists(
theme,
'link_default_text_decoration',
'--link-default-text-transform',
(v) => (v[2] ? 'uppercase' : 'none')
)
style.addIfExists(
theme,
'link_default_text_decoration',
'--link-default-font-style',
(v) => (v[3] ? 'italic' : 'none')
)
style.addIfExists(
theme,
'link_hover_text_decoration',
'--link-hover-text-decoration',
(v) => {
const value = []
if (v[0]) {
value.push('underline')
}
if (v[1]) {
value.push('line-through')
}
if (value.length === 0) {
return 'none'
}
return value.join(' ')
}
)
style.addIfExists(
theme,
'link_hover_text_decoration',
'--link-hover-text-transform',
(v) => (v[2] ? 'uppercase' : 'none')
)
style.addIfExists(
theme,
'link_hover_text_decoration',
'--link-hover-font-style',
(v) => (v[3] ? 'italic' : 'none')
)
style.addIfExists(
theme,
'link_active_text_decoration',
'--link-active-text-decoration',
(v) => {
const value = []
if (v[0]) {
value.push('underline')
}
if (v[1]) {
value.push('line-through')
}
if (value.length === 0) {
return 'none'
}
return value.join(' ')
}
)
style.addIfExists(
theme,
'link_active_text_decoration',
'--link-active-text-transform',
(v) => (v[2] ? 'uppercase' : 'none')
)
style.addIfExists(
theme,
'link_active_text_decoration',
'--link-active-font-style',
(v) => (v[3] ? 'italic' : 'none')
)
style.addPixelValueIfExists(theme, `link_font_size`)
style.addFontWeightIfExists(theme, `link_font_weight`)
return style.toObject()

View file

@ -41,3 +41,4 @@
@import 'auth_provider_with_modal';
@import 'side_bar';
@import 'custom_color_input';
@import 'text_decoration_selector';

View file

@ -8,6 +8,9 @@
font-weight: var(--heading-h1-font-weight, 700);
text-align: var(--heading-h1-text-alignment, left);
font-family: var(--heading-h1-font-family, Inter);
text-decoration: var(--heading-h1-text-decoration, none);
text-transform: var(--heading-h1-text-transform, none);
font-style: var(--heading-h1-font-style, none);
}
.ab-heading--h2 {
@ -16,6 +19,9 @@
font-weight: var(--heading-h2-font-weight, 600);
text-align: var(--heading-h2-text-alignment, left);
font-family: var(--heading-h2-font-family, Inter);
text-decoration: var(--heading-h2-text-decoration, none);
text-transform: var(--heading-h2-text-transform, none);
font-style: var(--heading-h2-font-style, none);
}
.ab-heading--h3 {
@ -24,6 +30,9 @@
font-weight: var(--heading-h3-font-weight, 500);
text-align: var(--heading-h3-text-alignment, left);
font-family: var(--heading-h3-font-family, Inter);
text-decoration: var(--heading-h3-text-decoration, none);
text-transform: var(--heading-h3-text-transform, none);
font-style: var(--heading-h3-font-style, none);
}
.ab-heading--h4 {
@ -32,6 +41,9 @@
font-weight: var(--heading-h4-font-weight, 500);
text-align: var(--heading-h4-text-alignment, left);
font-family: var(--heading-h4-font-family, Inter);
text-decoration: var(--heading-h4-text-decoration, none);
text-transform: var(--heading-h4-text-transform, none);
font-style: var(--heading-h4-font-style, none);
}
.ab-heading--h5 {
@ -40,6 +52,9 @@
font-weight: var(--heading-h5-font-weight, 400);
text-align: var(--heading-h5-text-alignment, left);
font-family: var(--heading-h5-font-family, Inter);
text-decoration: var(--heading-h5-text-decoration, none);
text-transform: var(--heading-h5-text-transform, none);
font-style: var(--heading-h5-font-style, none);
}
.ab-heading--h6 {
@ -48,5 +63,7 @@
font-weight: var(--heading-h6-font-weight, 400);
text-align: var(--heading-h6-text-alignment, left);
font-family: var(--heading-h6-font-family, Inter);
font-style: italic;
text-decoration: var(--heading-h6-text-decoration, underline);
text-transform: var(--heading-h6-text-transform, none);
font-style: var(--heading-h6-font-style, italic);
}

View file

@ -1,7 +1,9 @@
.ab-link {
font-size: var(--link-font-size, 14px);
font-weight: var(--link-font-weight, 400);
text-decoration: var(--link-text-decoration, underline);
text-decoration: var(--link-default-text-decoration, underline);
text-transform: var(--link-default-text-transform, none);
font-style: var(--link-default-font-style, none);
color: var(--link-text-color, $black);
align-self: var(--force-self-alignment, var(--link-text-alignment, initial));
font-family: var(--link-font-family, Inter);
@ -10,10 +12,14 @@
&--force-hover {
color: var(--link-hover-text-color, $black);
text-decoration: var(--link-hover-text-decoration, underline);
text-transform: var(--link-hover-text-transform, none);
font-style: var(--link-hover-font-style, none);
}
&--force-active {
color: var(--link-active-text-color, $black);
text-decoration: var(--link-active-text-decoration, underline);
text-transform: var(--link-active-text-transform, none);
font-style: var(--link-active-font-style, none);
}
}

View file

@ -3,6 +3,14 @@
flex-direction: column;
}
/**
Disable pointer events when in Page Editor.
*/
.element--read-only .ab-link {
pointer-events: none;
user-select: none;
}
.link-element__link {
font-size: 14px;
color: $black;

View file

@ -0,0 +1,4 @@
.text-decoration-selector {
display: flex;
gap: 2px;
}

View file

@ -21,6 +21,8 @@
overflow: hidden;
padding-right: 14px;
padding-top: 4px;
display: flex;
flex-direction: column;
.theme-config-block--no-preview & {
display: none;

View file

@ -0,0 +1,65 @@
<template>
<ButtonIcon
type="secondary"
v-bind="restProps"
:loading="loading"
:disabled="disabled"
:icon="icon"
:title="title"
:active="value"
@click.prevent="select()"
>
<slot></slot>
</ButtonIcon>
</template>
<script>
export default {
name: 'SwitchButton',
model: {
prop: 'modelValue',
event: 'input',
},
props: {
value: {
type: Boolean,
required: true,
default: false,
},
loading: {
type: Boolean,
required: false,
default: false,
},
disabled: {
type: Boolean,
required: false,
default: false,
},
icon: {
type: String,
required: false,
default: '',
},
title: {
type: String,
required: false,
default: '',
},
},
computed: {
restProps() {
const { value, modelValue, ...rest } = this.$attrs
return rest
},
},
methods: {
select() {
if (this.disabled) {
return
}
this.$emit('input', !this.value)
},
},
}
</script>

View file

@ -61,6 +61,7 @@ import RadioButton from '@baserow/modules/core/components/RadioButton'
import Thumbnail from '@baserow/modules/core/components/Thumbnail'
import ColorInput from '@baserow/modules/core/components/ColorInput'
import SelectSearch from '@baserow/modules/core/components/SelectSearch'
import SwitchButton from '@baserow/modules/core/components/SwitchButton'
function setupVue(Vue) {
Vue.component('Context', Context)
@ -111,6 +112,7 @@ function setupVue(Vue) {
Vue.component('ReadOnlyForm', ReadOnlyForm)
Vue.component('FormSection', FormSection)
Vue.component('SegmentControl', SegmentControl)
Vue.component('SwitchButton', SwitchButton)
Vue.filter('lowercase', lowercase)
Vue.filter('uppercase', uppercase)