Skip to content

Commit

Permalink
Squashed commit of the following:
Browse files Browse the repository at this point in the history
commit 11ee676
Author: Rebecca Graber <[email protected]>
Date:   Thu Jan 16 08:47:09 2025 -0500

    feat(projectHistoryLogs): log new submissions (#5416)

    ### 📣 Summary
    Create logs when new submissions are added to projects.

    ### 👷 Description for instance maintainers
    Allow null user_uids in AuditLogs so we can log anonymous submissions.

    ### 💭 Notes
    We previously had no need for null users in audit logs because the
    actions we logged were all restricted to authenticated users, but since
    we allow anonymous submissions, we needed a way to log those.

    ### 👀 Preview steps

    Feature/no-change template:
    1. ℹ️ have an account and a project. Make sure the account username is
    not `admin` (see [this notion
    task](https://www.notion.so/kobotoolbox/Anonymous-submissions-dont-work-if-user-named-admin-owns-asset-1767e515f65480608dfcee76ba9b3710?pvs=4))
    2. Deploy the project
    3. Add a submission to the project
    4. Go to `api/v2/asset/<asset-uid>/history`
    5. 🟢 There should be a new project history log with
    `action='add-submission'` and all the usual metadata, plus
    ```
    "submission": {
        "submitted_by": "user1"
    }
    ```
    6. Enable submissions without username/email to the project
    7. To make sure you're submitting anonymously, copy and paste the enketo
    link into a new private tab and add a new submission
    8. 🟢 Reload the endpoint. There should be a new audit log with
    `action='add-submission'`
    a. The user should be
    `http://kf.kobo.local:8080/api/v2/users/AnonymousUser/`
      b. The user_uid will be the uid of the anonymous user in the database
      c. The username should be `AnonymousUser`
    d. The metadata should contain `{"submission": {"submitted_by":
    "AnonymousUser"}` in addition to the usual

commit bbfdaf1
Author: olive-KTB <[email protected]>
Date:   Thu Jan 16 02:49:14 2025 +0100

    update gitlab-ci.yml

commit bc56f8f
Author: Akuukis <[email protected]>
Date:   Wed Jan 15 10:50:39 2025 +0200

    refactor(frontend): Mantine Component Library PoC (#5344)

    ### 💭 Notes

    Please read the PR commit-by-commit, here's a guide.

    1.
    [3105f47](3105f47)
    Setup Mantine Component Library. Please install the new dependencies,
    and add recommended VSCode extensions.
    2.
    [453f2c1](453f2c1)
    Here's a preview that Mantine in general works, and API for the example
    Button is different but generally similar.

    | our button | Mantine default button |
    |--------|--------|
    |
    ![image](https://github.com/user-attachments/assets/1f3e5736-c5eb-4cbf-a56e-a393dab561d3)
    |
    ![image](https://github.com/user-attachments/assets/c8f2a9dc-cf1a-44c3-881a-ff05da433060)
    |

    ![image](https://github.com/user-attachments/assets/e2abe407-3113-4900-9027-43186137ce13)

    3.
    [d782e38](d782e38)
    Example of custom styled component, Button. IMHO achieves pixel-perfect
    match in storybook and example above, except for line breaks, spinner
    and hover animations. Click animation matches out of box. Icon-only
    buttons are omitted because Mantine uses a different component
    `IconAction` for those.

    | Original on left / Mantine implementation on right |
    | --- |
    |
    ![image](https://github.com/user-attachments/assets/9a58b7af-3ff4-4b18-995c-6c57ff4264cb)
    |

    5.
    [eaf45e3](eaf45e3)
    Wrapped Button to add support for inbuilt Tooltip. No idea if we want to
    move forward with these two coupled, but I found it useful for
    comparison by re-implementing part of old Button behavior that's
    represented in storybook.

    | Original | Mantine implementation |
    | --- | --- |
    |
    ![image](https://github.com/user-attachments/assets/7bf2a504-902b-4801-a2dd-3e7648166316)
    |
    ![image](https://github.com/user-attachments/assets/41bc4dff-acf3-4328-ab6f-f7fcccbde20b)
    |

    ### 👀 Preview steps

    1. ℹ️ open Kobo home
    4. 🟢 [on main] notice the original "new" button
    6. 🟢 [on PR] notice the new "new" button with default mantine style,
    only slightly different

    ---------

    Co-authored-by: Leszek Pietrzak <[email protected]>
    Co-authored-by: Leszek <[email protected]>
    Co-authored-by: James Kiger <[email protected]>
    Co-authored-by: Paulo Amorim <[email protected]>
    Co-authored-by: James Kiger <[email protected]>

commit 9598180
Author: Raj Patel <[email protected]>
Date:   Wed Jan 15 13:29:03 2025 +0530

    fix!: reject duplicate submissions (#5047)

    ## Summary
    Implemented logic to detect and reject duplicate submissions.

    ## Description

    We have identified a race condition in the submission processing that
    causes duplicate submissions with identical UUIDs and XML hashes. This
    issue is particularly problematic under conditions with multiple remote
    devices submitting forms simultaneously over unreliable networks.

    To address this issue, a PR has been raised with the following proposed
    changes:

    - Race Condition Resolution: A locking mechanism has been added to
    prevent the race condition when checking for existing instances and
    creating new ones. This aims to eliminate duplicate submissions.

    - UUID Enforcement: Submissions without a UUID are now explicitly
    disallowed. This ensures that every submission is uniquely identifiable
    and further mitigates the risk of duplicate entries.

    - Introduction of `root_uuid`:

    - To ensure a consistent submission UUID throughout its lifecycle and
    prevent duplicate submissions with the same UUID, a new `root_uuid`
    column has been added to the `Instance` model with a unique constraint
    (`root_uuid` per `xform`).

    - If the `<meta><rootUuid>` is present in the submission XML, it is
    stored in the `root_uuid` column.

    - If `<meta><rootUuid>` is not present, the value from
    `<meta><instanceID>` is used instead.

    - This approach guarantees that the `root_uuid` remains constant across
    the lifecycle of a submission, providing a reliable identifier for all
    instances.

    - UUID Handling Improvement: Updated the logic to strip only the `uuid:`
    prefix while preserving custom, non-UUID ID schemes (e.g.,
    domain.com:1234). This ensures compliance with the OpenRosa spec and
    prevents potential ID collisions with custom prefixes.

    - Error Handling:
    - 202 Accepted: Returns when content is identical to an existing
    submission and successfully processed.
    - 409 Conflict: Returns when a duplicate UUID is detected but with
    differing content.

    These changes should improve the robustness of the submission process
    and prevent both race conditions and invalid submissions.

    ## Notes

    - Implemented a fix to address the race condition that leads to
    duplicate submissions with the same UUID and XML hash.
    - Incorporated improvements from existing work, ensuring consistency and
    robustness in handling concurrent submissions.
    - The fix aims to prevent duplicate submissions, even under high load
    and unreliable network conditions.

    ## Related issues

    Supersedes
    [kobotoolbox/kobocat#876](kobotoolbox/kobocat#876)
    and kobotoolbox/kobocat#859

    ---------

    Co-authored-by: Olivier Leger <[email protected]>

commit d22b8b5
Merge: 89bd9b7 b4aa1b7
Author: John N. Milner <[email protected]>
Date:   Tue Jan 14 15:20:49 2025 -0500

    Merge remote-tracking branch 'origin/release/2.024.36'

commit 89bd9b7
Author: Rebecca Graber <[email protected]>
Date:   Tue Jan 14 15:11:37 2025 -0500

    fix(auditLogs): correctly serialize audit logs from deleted users (#5418)

    ### 📣 Summary
    Fixes a 500 error from the various audit log endpoints when there are
    actions by deleted users.

    ### 📖 Description
    Return empty user and username fields in the response if the user was
    deleted after the log was created. This applies to `/api/v2/audit-logs`,
    `api/v2/assets/<uid>/history`, and `api/v2/project-history-logs`.

    ### 💭 Notes
    Small fix in the serializer. Also updates the ProjectHistoryLog
    serializer to inherit from the AuditLogSerializer so we don't have to
    duplicate the method fields.

    ### 👀 Preview steps

    Bug template:
    1. ℹ️ have a super user account and a project
    2. Create a new user (user1) and give them the `Edit Form` permission on
    the project.
    3. Log in as user1 and make an edit to the project.
    4. Log out user1 and log back in as the super user
    5. Delete user1. You can do this from the admin page if you delete the
    user from the User list, then from the Trash Bin.
    6. Go to:
      a. `api/v2/audit-logs`
      b. `api/v2/project-history-logs`
      c. `api/v2/assets/<uid>/history`
    7. 🔴 [on main] All will return a 500 error (`AttributeError: 'NoneType'
    object has no attribute 'username'`)
    8. 🟢 [on PR] The endpoint will return the expected logs. For all user1's
    actions, the user and username fields will be empty. but the user_uid
    should still refer to the old user.

commit b4aa1b7
Author: jnm <[email protected]>
Date:   Tue Jan 14 15:01:18 2025 -0500

    feat: export background-geopoint as GPS field (#5420)

    See kobotoolbox/formpack#327. This change just updates the formpack
    commit hash used by KPI

commit dddd619
Author: Olivier Léger <[email protected]>
Date:   Tue Jan 14 14:10:31 2025 -0500

    fix: catch additional XLSForm validation errors during deployment (#5419)

    ### 📣 Summary
    Enhanced error handling to catch more validation errors in XLSForm
    during deployment.

    ### 📖 Description
    Validation error handling for XLSForm deployment has been enhanced to
    catch a wider range of issues. This prevents the display of a generic
    500 error in the deployment modal and instead returns the explicit error
    message.

    ### Notes
    Supersedes #5417, #5411 and #5403

commit 9189ac9
Author: Olivier Léger <[email protected]>
Date:   Tue Jan 14 09:40:34 2025 -0500

    fix: handle case sensitivity for "Settings" sheet name with explicit error TASK-1353 (#5417)

    ### 📣 Summary
    Improved error handling for case-sensitive "Settings" sheet names.

    ### 📖 Description
    This update addresses an issue where a sheet named "Settings" with
    uppercase or mixed case letters causes unexpected behavior. An explicit
    error message is now raised to alert users of the case sensitivity,
    ensuring they can resolve the issue easily.

commit 8e8d6bb
Author: Rebecca Graber <[email protected]>
Date:   Mon Jan 13 15:02:04 2025 -0500

    test: rename admin user in fixture (#5415)

    ### 💭 Notes
    Developer-facing changes only. Changes the username of the admin user to
    `adminuser` in preparation for disallowing the name `admin` as part of
    https://www.notion.so/kobotoolbox/Anonymous-submissions-dont-work-if-user-named-admin-owns-asset-1767e515f65480608dfcee76ba9b3710
  • Loading branch information
magicznyleszek committed Jan 17, 2025
1 parent 14ffed8 commit 6353d2c
Show file tree
Hide file tree
Showing 93 changed files with 2,760 additions and 1,048 deletions.
1 change: 1 addition & 0 deletions .github/workflows/pytest.yml
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@ jobs:
KOBOCAT_URL: http://kobocat
KOBOCAT_INTERNAL_URL: http://kobocat
KOBOFORM_URL: http://kpi
SKIP_TESTS_WITH_CONCURRENCY: 'True'
strategy:
matrix:
python-version: ['3.10']
Expand Down
9 changes: 8 additions & 1 deletion .gitlab-ci.yml
Original file line number Diff line number Diff line change
Expand Up @@ -44,9 +44,16 @@ deploy-staging:
script:
- BRANCH_TITLE=${CI_COMMIT_BRANCH#feature/}
- helm repo add kobo https://gitlab.com/api/v4/projects/32216873/packages/helm/stable
- helm -n kobo-dev upgrade --install $BRANCH_TITLE kobo/kobo --atomic --set-string kpi.image.tag=${CI_COMMIT_SHORT_SHA} --reuse-values
- |
if [ "$CI_COMMIT_BRANCH" = "main" ]; then
helm -n kobo-dev upgrade staging-main kobo/kobo --atomic --set-string kpi.image.tag=${CI_COMMIT_SHORT_SHA} --reuse-values
helm -n kobo-dev upgrade staging-nobill kobo/kobo --atomic --set-string kpi.image.tag=${CI_COMMIT_SHORT_SHA} --reuse-values
else
helm -n kobo-dev upgrade --install $BRANCH_TITLE kobo/kobo --atomic --set-string kpi.image.tag=${CI_COMMIT_SHORT_SHA} --reuse-values
fi
rules:
- if: $CI_COMMIT_BRANCH =~ /^(feature\/)/ && $CI_COMMIT_REF_PROTECTED
- if: $CI_COMMIT_BRANCH == "main"

pages:
stage: deploy
Expand Down
9 changes: 7 additions & 2 deletions .storybook/main.js
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,12 @@ if (process.env.MEASURE) {
}
module.exports = {
stories: ['../jsapp/**/*.stories.@(js|jsx|ts|tsx)'],
addons: ['@storybook/addon-links', '@storybook/addon-essentials', '@storybook/addon-interactions', '@storybook/addon-a11y'
addons: [
'@storybook/addon-links',
'@storybook/addon-essentials',
'@storybook/addon-interactions',
'@storybook/addon-a11y',
'storybook-dark-mode',
// NB:
// 'storybook-addon-swc' may improve build speed in the future.
// - At time of writing, the build performance gains are negated because it
Expand Down Expand Up @@ -95,4 +100,4 @@ function applySpeedTweaks(config) {
terserOptions: {}
})];
}
}
}
32 changes: 32 additions & 0 deletions .storybook/preview.js
Original file line number Diff line number Diff line change
@@ -1,5 +1,37 @@
import 'jsapp/scss/main.scss';
import 'js/bemComponents';
import '@mantine/core/styles.css';
import {useEffect} from 'react';
import {addons} from '@storybook/preview-api';
import {DARK_MODE_EVENT_NAME} from 'storybook-dark-mode';
import {
MantineProvider,
useMantineColorScheme,
} from '@mantine/core';
import {themeKobo} from 'jsapp/js/theme';

const channel = addons.getChannel();

function ColorSchemeWrapper({children}) {
const {setColorScheme} = useMantineColorScheme();
const handleColorScheme = (value) => setColorScheme(value ? 'dark' : 'light');

useEffect(() => {
channel.on(DARK_MODE_EVENT_NAME, handleColorScheme);
return () => channel.off(DARK_MODE_EVENT_NAME, handleColorScheme);
}, [channel]);

return <>{children}</>;
}

export const decorators = [
(renderStory) => (
<ColorSchemeWrapper>{renderStory()}</ColorSchemeWrapper>
),
(renderStory) => (
<MantineProvider theme={themeKobo}>{renderStory()}</MantineProvider>
),
];

export const parameters = {
options: {
Expand Down
2 changes: 1 addition & 1 deletion dependencies/pip/dev_requirements.txt
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@
# via -r dependencies/pip/requirements.in
-e git+https://github.com/trevoriancox/django-dont-vary-on.git@01a804122b7ddcdc22f50b40993f91c27b03bef6#egg=django-dont-vary-on
# via -r dependencies/pip/requirements.in
-e git+https://github.com/kobotoolbox/formpack.git@5a8cda8cc37a56a2313c98b88fcacc18049ef477#egg=formpack
-e git+https://github.com/kobotoolbox/formpack.git@894d285b1d2955b2990500427158e845af32259d#egg=formpack
# via -r dependencies/pip/requirements.in
-e git+https://github.com/dimagi/python-digest@5c94bb74516b977b60180ee832765c0695ff2b56#egg=python_digest
# via -r dependencies/pip/requirements.in
Expand Down
2 changes: 1 addition & 1 deletion dependencies/pip/requirements.in
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@
# https://github.com/bndr/pipreqs is a handy utility, too.

# formpack
-e git+https://github.com/kobotoolbox/formpack.git@5a8cda8cc37a56a2313c98b88fcacc18049ef477#egg=formpack
-e git+https://github.com/kobotoolbox/formpack.git@894d285b1d2955b2990500427158e845af32259d#egg=formpack

# More up-to-date version of django-digest than PyPI seems to have.
# Also, python-digest is an unlisted dependency thereof.
Expand Down
2 changes: 1 addition & 1 deletion dependencies/pip/requirements.txt
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@
# via -r dependencies/pip/requirements.in
-e git+https://github.com/trevoriancox/django-dont-vary-on.git@01a804122b7ddcdc22f50b40993f91c27b03bef6#egg=django-dont-vary-on
# via -r dependencies/pip/requirements.in
-e git+https://github.com/kobotoolbox/formpack.git@5a8cda8cc37a56a2313c98b88fcacc18049ef477#egg=formpack
-e git+https://github.com/kobotoolbox/formpack.git@894d285b1d2955b2990500427158e845af32259d#egg=formpack
# via -r dependencies/pip/requirements.in
-e git+https://github.com/dimagi/python-digest@5c94bb74516b977b60180ee832765c0695ff2b56#egg=python_digest
# via -r dependencies/pip/requirements.in
Expand Down
75 changes: 40 additions & 35 deletions jsapp/js/app.jsx
Original file line number Diff line number Diff line change
Expand Up @@ -29,11 +29,14 @@ import {
import {isAnyProcessingRouteActive} from 'js/components/processing/routes.utils';
import pageState from 'js/pageState.store';

import '@mantine/core/styles.css';
import { MantineProvider } from '@mantine/core';

// Query-related
import { QueryClientProvider } from '@tanstack/react-query';
import { ReactQueryDevtools } from '@tanstack/react-query-devtools';
import { queryClient } from './query/queryClient.ts';
import { RequireOrg } from './router/RequireOrg';
import { themeKobo } from './theme';

class App extends React.Component {
constructor(props) {
Expand Down Expand Up @@ -107,46 +110,48 @@ class App extends React.Component {
return (
<DocumentTitle title='KoboToolbox'>
<QueryClientProvider client={queryClient}>
<RootContextProvider>
<RequireOrg>
<Tracking />
<ToasterConfig />

{this.shouldDisplayMainLayoutElements() &&
<div className='header-stretch-bg' />
}

<bem.PageWrapper
m={pageWrapperModifiers}
className='mdl-layout mdl-layout--fixed-header'
>
{this.state.pageState.modal && (
<BigModal params={this.state.pageState.modal} />
)}

{this.shouldDisplayMainLayoutElements() && (
<>
<MainHeader assetUid={assetid} />
<Drawer />
</>
)}

<bem.PageWrapper__content
className='mdl-layout__content'
m={pageWrapperContentModifiers}
<MantineProvider theme={themeKobo}>
<RootContextProvider>
<RequireOrg>
<Tracking />
<ToasterConfig />

{this.shouldDisplayMainLayoutElements() &&
<div className='header-stretch-bg' />
}

<bem.PageWrapper
m={pageWrapperModifiers}
className='mdl-layout mdl-layout--fixed-header'
>
{this.state.pageState.modal && (
<BigModal params={this.state.pageState.modal} />
)}

{this.shouldDisplayMainLayoutElements() && (
<>
{this.isFormSingle() && <ProjectTopTabs />}
<FormViewSideTabs show={this.isFormSingle()} />
<MainHeader assetUid={assetid} />
<Drawer />
</>
)}

<Outlet />
</bem.PageWrapper__content>
</bem.PageWrapper>
</RequireOrg>
</RootContextProvider>
<bem.PageWrapper__content
className='mdl-layout__content'
m={pageWrapperContentModifiers}
>
{this.shouldDisplayMainLayoutElements() && (
<>
{this.isFormSingle() && <ProjectTopTabs />}
<FormViewSideTabs show={this.isFormSingle()} />
</>
)}

<Outlet />
</bem.PageWrapper__content>
</bem.PageWrapper>
</RequireOrg>
</RootContextProvider>
</MantineProvider>


{/* React Query Devtools - GUI for inspecting and modifying query status
Expand Down
9 changes: 0 additions & 9 deletions jsapp/js/bemComponents.ts
Original file line number Diff line number Diff line change
Expand Up @@ -179,15 +179,6 @@ bem.KDrawer__primaryIcons = makeBem(bem.KDrawer, 'primary-icons', 'nav');
bem.KDrawer__secondaryIcons = makeBem(bem.KDrawer, 'secondary-icons', 'nav');
bem.KDrawer__sidebar = makeBem(bem.KDrawer, 'sidebar', 'aside');

bem.SimpleTable = makeBem(null, 'simple-table', 'table');
bem.SimpleTable__header = makeBem(bem.SimpleTable, 'header', 'thead');
bem.SimpleTable__body = makeBem(bem.SimpleTable, 'body', 'tbody');
bem.SimpleTable__footer = makeBem(bem.SimpleTable, 'footer', 'tfoot');
bem.SimpleTable__row = makeBem(bem.SimpleTable, 'row', 'tr');
// NOTE: messageRow needs a __cell with colspan set
bem.SimpleTable__messageRow = makeBem(bem.SimpleTable, 'message-row', 'tr');
bem.SimpleTable__cell = makeBem(bem.SimpleTable, 'cell', 'td');

bem.tagSelect = makeBem(null, 'tag-select');
bem.collectionFilter = makeBem(null, 'collection-filter');

Expand Down
27 changes: 27 additions & 0 deletions jsapp/js/components/common/ButtonNew.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
import {Button as ButtonMantine, createPolymorphicComponent, Tooltip} from '@mantine/core';
import type {ButtonProps as ButtonPropsMantine, TooltipProps} from '@mantine/core/lib/components';
import {forwardRef} from 'react';

// See boilerpate at: https://mantine.dev/guides/polymorphic/#wrapping-polymorphic-components

export interface ButtonProps extends ButtonPropsMantine {
tooltip?: React.ReactNode;
tooltipProps?: Partial<Omit<TooltipProps, 'label'>>;
}

const Button = forwardRef<HTMLButtonElement, ButtonProps>(({tooltip, tooltipProps, ...others}, ref) => {
if (!tooltip) {
return (
<ButtonMantine {...others} ref={ref} />
);
}

return (
<Tooltip label={tooltip} {...tooltipProps}>
<ButtonMantine {...others} ref={ref} />
</Tooltip>
);
});
Button.displayName = 'Button';

export default createPolymorphicComponent<'button', ButtonProps>(Button);
25 changes: 25 additions & 0 deletions jsapp/js/components/common/SimpleTable.module.scss
Original file line number Diff line number Diff line change
@@ -0,0 +1,25 @@
// We need bigger specificity in each selector to ensure we do overwrite default
// styles

table.SimpleTableRoot {
background-color: var(--mantine-color-gray-9);
border-collapse: separate;
border-radius: var(--mantine-radius-md);
}

thead.SimpleTableThead {
background-color: var(--mantine-color-gray-8);
}

th.SimpleTableTh {
font-size: var(--mantine-font-size-sm);
color: var(--mantine-color-gray-2);
font-weight: 400;
}

td.SimpleTableTd {
font-size: var(--mantine-font-size-md);
border-top-width: 1px;
border-top-color: var(--mantine-color-gray-7);
border-top-style: solid;
}
46 changes: 46 additions & 0 deletions jsapp/js/components/common/SimpleTable.stories.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,46 @@
import type {Meta, StoryObj} from '@storybook/react';
import SimpleTable from './SimpleTable';

const meta: Meta<React.ComponentProps<typeof SimpleTable>> = {
title: 'common/SimpleTable',
component: SimpleTable,
argTypes: {},
args: {},
render: ({...args}) => (
<SimpleTable
{...args}
head={['Element position', 'Atomic mass', 'Symbol', 'Element name']}
body={
[
[6, 12.011, 'C', 'Carbon'],
[7, 14.007, 'N', 'Nitrogen'],
[39, 88.906, 'Y', 'Yttrium'],
[56, 137.33, 'Ba', 'Barium'],
[
'n/a',
'n/a',
'??',
(
<div key='test'>
This is just a DIV. It has a button and an input:
<br/><br/>
<button>button</button>
<br/><br/>
<input type='email'/>
<br/><br/>
It shows you can have any <code>React.ReactNode</code> here.
</div>
),
],
[58, 140.12, 'Ce', 'Cerium'],
]
}
/>
),
};

export default meta;

export const Primary: StoryObj<typeof SimpleTable> = {
args: {},
};
47 changes: 47 additions & 0 deletions jsapp/js/components/common/SimpleTable.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,47 @@
import {Table, type MantineStyleProps, type TableData} from '@mantine/core';
import styles from './SimpleTable.module.scss';

interface SimpleTableProps extends MantineStyleProps {
head: TableData['head'];
body: TableData['body'];
/**
* Passing minimum width enables contextual horizontal scrollbar (i.e. without
* it the table will never display scrollbar - regardless of how small
* the screen is).
*/
minWidth?: number;
}

/**
* A wrapper component for `Table` from `@mantine/core`. It requires column
* headings, column data, and has optional minimum width. You can pass all
* standard Mantine style props down to the inner `Table`.
*/
export default function SimpleTable(
{head, body, minWidth, ...styleProps}: SimpleTableProps
) {
const table = (
<Table
{...styleProps}
classNames={{
table: styles.SimpleTableRoot,
thead: styles.SimpleTableThead,
th: styles.SimpleTableTh,
td: styles.SimpleTableTd,
}}
data={{head: head, body: body}}
horizontalSpacing='sm'
verticalSpacing='sm'
/>
);

if (minWidth) {
return (
<Table.ScrollContainer minWidth={minWidth} type='native'>
{table}
</Table.ScrollContainer>
);
}

return table;
}
Loading

0 comments on commit 6353d2c

Please sign in to comment.