1
0
Fork 0
mirror of https://gitlab.com/bramw/baserow.git synced 2025-04-03 04:35:31 +00:00

Resolve "Make the table element responsive"

This commit is contained in:
Andrea Disarò 2024-06-13 12:45:52 +00:00 committed by Afonso Silva
parent ad5da1f276
commit d62e4fdb46
16 changed files with 320 additions and 82 deletions
backend
src/baserow/contrib/builder
tests/baserow/contrib/builder
changelog/entries/unreleased/feature
web-frontend/modules

View file

@ -42,6 +42,7 @@ from baserow.contrib.builder.elements.models import (
TableElement,
TextElement,
VerticalAlignments,
get_default_table_orientation,
)
from baserow.contrib.builder.elements.registries import (
ElementType,
@ -219,14 +220,15 @@ class TableElementType(CollectionElementWithFieldsTypeMixin, ElementType):
class SerializedDict(CollectionElementWithFieldsTypeMixin.SerializedDict):
button_color: str
orientation: dict
@property
def allowed_fields(self):
return super().allowed_fields + ["button_color"]
return super().allowed_fields + ["button_color", "orientation"]
@property
def serializer_field_names(self):
return super().serializer_field_names + ["button_color"]
return super().serializer_field_names + ["button_color", "orientation"]
@property
def serializer_field_overrides(self):
@ -238,10 +240,18 @@ class TableElementType(CollectionElementWithFieldsTypeMixin, ElementType):
default="primary",
help_text="Button color.",
),
"orientation": serializers.JSONField(
allow_null=False,
default=get_default_table_orientation,
help_text=TableElement._meta.get_field("orientation").help_text,
),
}
def get_pytest_params(self, pytest_data_fixture) -> Dict[str, Any]:
return {"data_source_id": None}
return {
"data_source_id": None,
"orientation": get_default_table_orientation(),
}
class RepeatElementType(

View file

@ -59,6 +59,14 @@ def get_default_element_content_type():
return ContentType.objects.get_for_model(Element)
def get_default_table_orientation():
return {
"smartphone": "horizontal",
"tablet": "horizontal",
"desktop": "horizontal",
}
class Element(
HierarchicalModelMixin,
TrashableModelMixin,
@ -765,7 +773,12 @@ class TableElement(CollectionElement):
blank=True,
help_text="The color of the button",
)
orientation = models.JSONField(
blank=True,
null=True,
default=get_default_table_orientation,
help_text="The table orientation (horizontal or vertical) for each device type",
)
fields = models.ManyToManyField(CollectionField)

View file

@ -0,0 +1,39 @@
# Generated by Django 4.1.13 on 2024-06-06 11:12
from django.db import migrations, models
from baserow.contrib.builder.elements.models import get_default_table_orientation
def populate_orientation_field(apps, schema_editor):
"""Add default orientation settings to all table elements."""
TableElement = apps.get_model("builder", "tableelement")
TableElement.objects.update(orientation={
"smartphone": "horizontal",
"tablet": "horizontal",
"desktop": "horizontal",
})
class Migration(migrations.Migration):
dependencies = [
("builder", "0022_choiceelement_choiceelementoption_and_more"),
]
operations = [
migrations.AddField(
model_name="tableelement",
name="orientation",
field=models.JSONField(
blank=True,
default=get_default_table_orientation,
help_text="The table orientation (horizontal or vertical) for each device type",
null=True,
),
),
migrations.RunPython(
populate_orientation_field,
reverse_code=migrations.RunPython.noop,
),
]

View file

@ -372,6 +372,11 @@ def test_builder_application_export(data_fixture):
"type": "table",
"order": str(element4.order),
"button_color": "primary",
"orientation": {
"smartphone": "horizontal",
"tablet": "horizontal",
"desktop": "horizontal",
},
"parent_element_id": None,
"place_in_container": None,
"visibility": "all",

View file

@ -0,0 +1,7 @@
{
"type": "feature",
"message": "[Builder] Add responsive behavior configuration for table element",
"issue_number": 2523,
"bullet_points": [],
"created_at": "2024-06-05"
}

View file

@ -1,47 +1,93 @@
<template>
<table class="baserow-table">
<thead>
<tr class="baserow-table__header-row">
<th
v-for="field in fields"
:key="field.__id__"
class="baserow-table__header-cell"
>
<slot name="field-name" :field="field">{{ field.name }}</slot>
</th>
</tr>
</thead>
<tbody v-if="rows.length">
<tr
v-for="(row, index) in rows"
:key="row.__id__"
class="baserow-table__row"
>
<td v-for="field in fields" :key="field.id" class="baserow-table__cell">
<slot
name="cell-content"
:value="row[field.name]"
:field="field"
:row-index="index"
<div class="baserow-table-wrapper">
<table class="baserow-table" :class="`baserow-table--${orientation}`">
<template v-if="orientation === TABLE_ORIENTATION.HORIZONTAL">
<thead>
<tr class="baserow-table__row">
<th
v-for="field in fields"
:key="field.__id__"
class="baserow-table__header-cell"
>
<slot name="field-name" :field="field">{{ field.name }}</slot>
</th>
</tr>
</thead>
<tbody v-if="rows.length">
<tr
v-for="(row, index) in rows"
:key="row.__id__"
class="baserow-table__row"
>
{{ value }}
</slot>
</td>
</tr>
</tbody>
<tbody v-else>
<tr>
<td class="baserow-table__empty-message" :colspan="fields.length">
<slot name="empty-state"></slot>
</td>
</tr>
</tbody>
</table>
<td
v-for="field in fields"
:key="field.id"
class="baserow-table__cell"
>
<slot
name="cell-content"
:value="row[field.name]"
:field="field"
:row-index="index"
>
{{ value }}
</slot>
</td>
</tr>
</tbody>
</template>
<template v-else>
<tbody v-if="rows.length">
<template v-for="(row, rowIndex) in rows">
<tr
v-for="(field, fieldIndex) in fields"
:key="`${row.__id__}_${field.id}`"
class="baserow-table__row"
>
<th
class="baserow-table__header-cell"
:class="{
'baserow-table__separator': fieldIndex === fields.length - 1,
}"
>
{{ field.name }}
</th>
<td
class="baserow-table__cell"
:class="{
'baserow-table__separator': fieldIndex === fields.length - 1,
}"
>
<slot
name="cell-content"
:value="row[field.name]"
:field="field"
:row-index="rowIndex"
>
{{ value }}
</slot>
</td>
</tr>
</template>
</tbody>
</template>
<tbody v-if="!rows.length">
<tr>
<td class="baserow-table__empty-message" :colspan="fields.length">
<slot name="empty-state"></slot>
</td>
</tr>
</tbody>
</table>
</div>
</template>
<script>
import { TABLE_ORIENTATION } from '@baserow/modules/builder/enums'
export default {
name: 'BaserowTable',
inject: ['mode'],
props: {
fields: {
type: Array,
@ -51,12 +97,18 @@ export default {
type: Array,
required: true,
},
orientation: {
type: String,
default: TABLE_ORIENTATION.HORIZONTAL,
},
},
data() {
return {}
},
computed: {},
watch: {},
methods: {},
computed: {
TABLE_ORIENTATION() {
return TABLE_ORIENTATION
},
},
}
</script>

View file

@ -5,7 +5,11 @@
}"
class="table-element"
>
<BaserowTable :fields="element.fields" :rows="rows">
<BaserowTable
:fields="element.fields"
:rows="rows"
:orientation="orientation"
>
<template #cell-content="{ rowIndex, field, value }">
<component
:is="collectionFieldTypes[field.type].component"
@ -55,6 +59,8 @@ export default {
* display.
* @property {Object} fields - The fields of the data source.
* @property {int} items_per_page - The number of items per page.
* @property {string} button_color - The color of the button.
* @property {string} orientation - The orientation for eaceh device.
*/
element: {
type: Object,
@ -94,6 +100,10 @@ export default {
collectionFieldTypes() {
return this.$registry.getAll('collectionField')
},
orientation() {
const device = this.$store.getters['page/getDeviceTypeSelected']
return this.element.orientation[device]
},
},
methods: {

View file

@ -30,10 +30,18 @@
@blur="$v.values.items_per_page.$touch()"
></FormInput>
<FormGroup :label="$t('repeatElementForm.orientationLabel')">
<RadioButton v-model="values.orientation" value="vertical">
<RadioButton
v-model="values.orientation"
value="vertical"
icon="iconoir-table-rows"
>
{{ $t('repeatElementForm.orientationVertical') }}
</RadioButton>
<RadioButton v-model="values.orientation" value="horizontal">
<RadioButton
v-model="values.orientation"
value="horizontal"
icon="iconoir-view-columns-3"
>
{{ $t('repeatElementForm.orientationHorizontal') }}
</RadioButton>
</FormGroup>

View file

@ -145,6 +145,30 @@
</template>
<p v-else>{{ $t('tableElementForm.selectSourceFirst') }}</p>
</FormGroup>
<FormGroup :label="$t('tableElementForm.orientation')">
<DeviceSelector
:device-type-selected="deviceTypeSelected"
direction="row"
@selected="actionSetDeviceTypeSelected"
>
<template #deviceTypeControl="{ deviceType }">
<RadioButton
v-model="values.orientation[deviceType.getType()]"
icon="iconoir-view-columns-3"
:value="TABLE_ORIENTATION.HORIZONTAL"
>
{{ $t('tableElementForm.orientationHorizontal') }}
</RadioButton>
<RadioButton
v-model="values.orientation[deviceType.getType()]"
icon="iconoir-table-rows"
:value="TABLE_ORIENTATION.VERTICAL"
>
{{ $t('tableElementForm.orientationVertical') }}
</RadioButton>
</template>
</DeviceSelector>
</FormGroup>
<ColorInputGroup
v-model="values.button_color"
:label="$t('tableElementForm.buttonColor')"
@ -168,23 +192,38 @@ import {
} from 'vuelidate/lib/validators'
import elementForm from '@baserow/modules/builder/mixins/elementForm'
import collectionElementForm from '@baserow/modules/builder/mixins/collectionElementForm'
import { TABLE_ORIENTATION } from '@baserow/modules/builder/enums'
import DeviceSelector from '@baserow/modules/builder/components/page/header/DeviceSelector.vue'
import { mapActions, mapGetters } from 'vuex'
export default {
name: 'TableElementForm',
components: { ApplicationBuilderFormulaInputGroup },
components: { DeviceSelector, ApplicationBuilderFormulaInputGroup },
mixins: [elementForm, collectionElementForm],
data() {
return {
allowedValues: ['data_source_id', 'fields', 'items_per_page'],
allowedValues: [
'data_source_id',
'fields',
'items_per_page',
'button_color',
'orientation',
],
values: {
fields: [],
data_source_id: null,
items_per_page: 1,
button_color: '',
orientation: {},
},
userHasChangedDataSource: false,
}
},
computed: {
...mapGetters({ deviceTypeSelected: 'page/getDeviceTypeSelected' }),
TABLE_ORIENTATION() {
return TABLE_ORIENTATION
},
orderedCollectionTypes() {
return this.$registry.getOrderedList('collectionField')
},
@ -203,6 +242,9 @@ export default {
},
},
methods: {
...mapActions({
actionSetDeviceTypeSelected: 'page/setDeviceTypeSelected',
}),
addField() {
this.values.fields.push({
name: getNextAvailableNameInSequence(

View file

@ -1,23 +1,22 @@
<template>
<ul class="header__filter">
<li
<div class="device-selector">
<div
v-for="(deviceType, index) in deviceTypes"
:key="deviceType.getType()"
class="header__filter-item"
:class="{ 'header__filter-item--no-margin-left': index === 0 }"
:key="index"
class="device-selector__item"
:class="`device-selector__item--${direction}`"
>
<a
class="header__filter-link"
:class="{
'active active--primary': deviceTypeSelected === deviceType.getType(),
}"
@click="$emit('selected', deviceType.getType())"
>
<i :class="`header__filter-icon ${deviceType.iconClass}`"></i>
</a>
<RadioButton
:key="deviceType.getType()"
:value="deviceType.getType()"
:icon="deviceType.iconClass"
:model-value="deviceTypeSelected"
class="device-selector__button"
@click.native="$emit('selected', deviceType.getType())"
></RadioButton>
<slot name="deviceTypeControl" :device-type="deviceType"></slot>
</li>
</ul>
</div>
</div>
</template>
<script>
@ -28,6 +27,11 @@ export default {
type: String,
required: true,
},
direction: {
type: String,
required: false,
default: 'column',
},
},
computed: {
deviceTypes() {

View file

@ -147,3 +147,8 @@ export const ELEMENT_EVENTS = {
DATA_SOURCE_REMOVED: 'DATA_SOURCE_REMOVED',
DATA_SOURCE_AFTER_UPDATE: 'DATA_SOURCE_AFTER_UPDATE',
}
export const TABLE_ORIENTATION = {
HORIZONTAL: 'horizontal',
VERTICAL: 'vertical',
}

View file

@ -427,7 +427,10 @@
"itemsPerPagePlaceholder": "Enter value...",
"selectSourceFirst": "Choose a list data source to begin configuring your fields.",
"buttonColor": "Button color",
"refreshFieldsFromDataSource": "refresh fields from data source"
"refreshFieldsFromDataSource": "refresh fields from data source",
"orientation": "Orientation",
"orientationHorizontal": "Horizontal",
"orientationVertical": "Vertical"
},
"tableElement": {
"empty": "No items have been found.",
@ -443,7 +446,7 @@
"itemsPerPage": "Items per page",
"itemsPerPagePlaceholder": "Enter value...",
"itemsPerRowLabel": "Items per row",
"itemsPerRowDescription": "Choose, per device type, what the number of repetitions per row should be.",
"itemsPerRowDescription": "Number of columns per row and device type.",
"orientationLabel": "Orientation",
"orientationVertical": "Vertical",
"orientationHorizontal": "Horizontal"

View file

@ -29,3 +29,4 @@
@import 'loading_spinner';
@import 'update_user_source_form';
@import 'user_source_users_context';
@import 'device_selector';

View file

@ -0,0 +1,31 @@
.device-selector {
display: flex;
flex-flow: row wrap;
align-items: center;
gap: 5px;
}
.device-selector__item {
display: flex;
gap: 5px;
}
.device-selector__item--column {
flex: 1;
flex-flow: column nowrap;
}
.device-selector__item--row {
flex-direction: row;
flex-grow: 1;
align-items: center;
}
.device-selector__button {
border: none;
box-shadow: none;
.button__icon {
margin: 0 auto;
}
}

View file

@ -1,3 +1,7 @@
.baserow-table-wrapper {
overflow-x: auto;
}
.baserow-table {
width: 100%;
border: 1px solid $black;
@ -6,10 +10,6 @@
font-size: 12px;
}
.baserow-table__header-row {
background-color: $palette-neutral-200;
}
.baserow-table__header-cell,
.baserow-table__cell {
padding: 12px 20px 10px;
@ -17,13 +17,28 @@
.baserow-table__header-cell {
font-weight: 600;
border-bottom: 1px solid $black;
background-color: $palette-neutral-200;
}
.baserow-table__cell {
border-bottom: 1px solid $black;
.baserow-table__separator {
.baserow-table__row:not(:last-child) & {
border-bottom: 1px solid $black;
}
}
.baserow-table__row:last-child & {
.baserow-table--horizontal {
.baserow-table__header-cell,
.baserow-table__cell {
border-bottom: 1px solid $black;
}
.baserow-table__row:last-child .baserow-table__cell {
border-bottom: none;
}
}
.baserow-table--vertical {
.baserow-table__header-cell {
border-right: 1px solid $black;
}
}

View file

@ -1,14 +1,7 @@
.repeat-element__device-selector {
gap: 10px;
input[type='number'] {
margin-top: 5px;
text-align: center;
}
.header__filter-icon {
margin: 0 auto;
}
}
.repeat-element__preview {