mirror of
https://github.com/inventree/InvenTree
synced 2024-08-30 18:33:04 +00:00
Forms initial data (#6365)
* Use PATCH for edit form * Add "localhost:8000" server option * Add initialData property for forms - Allows user to specify an initial dataset to override default values * Override field values when constructing form props * Remove debug messages * Wrap ApiForm in FormProvider - Allows lower elements to access all form data without rebuild - Pass all form data through to adjustFilters routine * Fixes for RelatedModelField - Ensure that the saved data are cleared when filters change * Fix debug message for token creation * Fix address rendering for modals * Refactor "address" forms - Use new "hook" form structure * Update Contacts table * Prevent related model fields from fetching on initial form open - Only fetch when the drop-down is actually opened - Significantly reduces the number of API calls * Fix for ChoiceField - Display label / description / placeholder text * Fix for DateInput - Correct conversion of datatype * Implement "new purchase order" form - Uses modal form hook - Supply initial data - Adjust filters according to selected company * Add new company from company table * Edit company from detail page * More table updates - StockLocation - PartCategory * Update more tables / forms: - PartParameter table - PartParameterTemplate table - Cleanup unused imports * Update ProjectCode table * CustomUnits table * Update RelatedPart table * Update PartTestTemplate table * Cleanup PartParameterTable * Add "IPN" column to PartParameterTable * Update BuildOrder table * Update BuildDetail page * PurchaseOrderLineItem table * Simplify - Move fields which are only used in one location * Create new salesorder with context - Also consolidate translated strings - Also improve consistency of inline rendering (with missing image) * Revert change to RenderInlineModel * Fix for build table - Use apiUrl wrapper around ApiEndpoint * Fix parameter for PurchaseOrderTable * Adjust server selector - Only show localhost:8000 if in dev mode * Tweak URL * Add extra test to playground - Check initial value works for nested field * Cleanup playground * Cleanup unused vars * memoize fields * Fix typo host -> host * Fix part editing * Cleanup unused * update group table
This commit is contained in:
parent
2557383892
commit
7fe8207463
@ -253,6 +253,12 @@ class GetAuthToken(APIView):
|
|||||||
# User is authenticated, and requesting a token against the provided name.
|
# User is authenticated, and requesting a token against the provided name.
|
||||||
token = ApiToken.objects.create(user=request.user, name=name)
|
token = ApiToken.objects.create(user=request.user, name=name)
|
||||||
|
|
||||||
|
logger.info(
|
||||||
|
"Created new API token for user '%s' (name='%s')",
|
||||||
|
user.username,
|
||||||
|
name,
|
||||||
|
)
|
||||||
|
|
||||||
# Add some metadata about the request
|
# Add some metadata about the request
|
||||||
token.set_metadata('user_agent', request.META.get('HTTP_USER_AGENT', ''))
|
token.set_metadata('user_agent', request.META.get('HTTP_USER_AGENT', ''))
|
||||||
token.set_metadata('remote_addr', request.META.get('REMOTE_ADDR', ''))
|
token.set_metadata('remote_addr', request.META.get('REMOTE_ADDR', ''))
|
||||||
@ -263,10 +269,6 @@ class GetAuthToken(APIView):
|
|||||||
|
|
||||||
data = {'token': token.key, 'name': token.name, 'expiry': token.expiry}
|
data = {'token': token.key, 'name': token.name, 'expiry': token.expiry}
|
||||||
|
|
||||||
logger.info(
|
|
||||||
"Created new API token for user '%s' (name='%s')", user.username, name
|
|
||||||
)
|
|
||||||
|
|
||||||
# Ensure that the users session is logged in (PUI -> CUI login)
|
# Ensure that the users session is logged in (PUI -> CUI login)
|
||||||
if not get_user(request).is_authenticated:
|
if not get_user(request).is_authenticated:
|
||||||
login(request, user)
|
login(request, user)
|
||||||
|
@ -14,6 +14,7 @@ import { useCallback, useEffect, useMemo } from 'react';
|
|||||||
import { useState } from 'react';
|
import { useState } from 'react';
|
||||||
import {
|
import {
|
||||||
FieldValues,
|
FieldValues,
|
||||||
|
FormProvider,
|
||||||
SubmitErrorHandler,
|
SubmitErrorHandler,
|
||||||
SubmitHandler,
|
SubmitHandler,
|
||||||
useForm
|
useForm
|
||||||
@ -65,6 +66,7 @@ export interface ApiFormProps {
|
|||||||
pathParams?: PathParams;
|
pathParams?: PathParams;
|
||||||
method?: 'GET' | 'POST' | 'PUT' | 'PATCH' | 'DELETE';
|
method?: 'GET' | 'POST' | 'PUT' | 'PATCH' | 'DELETE';
|
||||||
fields?: ApiFormFieldSet;
|
fields?: ApiFormFieldSet;
|
||||||
|
initialData?: FieldValues;
|
||||||
submitText?: string;
|
submitText?: string;
|
||||||
submitColor?: string;
|
submitColor?: string;
|
||||||
fetchInitialData?: boolean;
|
fetchInitialData?: boolean;
|
||||||
@ -146,6 +148,13 @@ export function OptionsApiForm({
|
|||||||
field: v,
|
field: v,
|
||||||
definition: data?.[k]
|
definition: data?.[k]
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// If the user has specified initial data, use that value here
|
||||||
|
let value = _props?.initialData?.[k];
|
||||||
|
|
||||||
|
if (value) {
|
||||||
|
_props.fields[k].value = value;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return _props;
|
return _props;
|
||||||
@ -163,13 +172,23 @@ export function OptionsApiForm({
|
|||||||
* based on an API endpoint.
|
* based on an API endpoint.
|
||||||
*/
|
*/
|
||||||
export function ApiForm({ id, props }: { id: string; props: ApiFormProps }) {
|
export function ApiForm({ id, props }: { id: string; props: ApiFormProps }) {
|
||||||
const defaultValues: FieldValues = useMemo(
|
const defaultValues: FieldValues = useMemo(() => {
|
||||||
() =>
|
let defaultValuesMap = mapFields(props.fields ?? {}, (_path, field) => {
|
||||||
mapFields(props.fields ?? {}, (_path, field) => {
|
return field.value ?? field.default ?? undefined;
|
||||||
return field.default ?? undefined;
|
});
|
||||||
}),
|
|
||||||
[props.fields]
|
// If the user has specified initial data, use that instead
|
||||||
);
|
if (props.initialData) {
|
||||||
|
defaultValuesMap = {
|
||||||
|
...defaultValuesMap,
|
||||||
|
...props.initialData
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update the form values, but only for the fields specified for this form
|
||||||
|
|
||||||
|
return defaultValuesMap;
|
||||||
|
}, [props.fields, props.initialData]);
|
||||||
|
|
||||||
// Form errors which are not associated with a specific field
|
// Form errors which are not associated with a specific field
|
||||||
const [nonFieldErrors, setNonFieldErrors] = useState<string[]>([]);
|
const [nonFieldErrors, setNonFieldErrors] = useState<string[]>([]);
|
||||||
@ -179,6 +198,7 @@ export function ApiForm({ id, props }: { id: string; props: ApiFormProps }) {
|
|||||||
criteriaMode: 'all',
|
criteriaMode: 'all',
|
||||||
defaultValues
|
defaultValues
|
||||||
});
|
});
|
||||||
|
|
||||||
const {
|
const {
|
||||||
isValid,
|
isValid,
|
||||||
isDirty,
|
isDirty,
|
||||||
@ -390,6 +410,7 @@ export function ApiForm({ id, props }: { id: string; props: ApiFormProps }) {
|
|||||||
{props.preFormWarning}
|
{props.preFormWarning}
|
||||||
</Alert>
|
</Alert>
|
||||||
)}
|
)}
|
||||||
|
<FormProvider {...form}>
|
||||||
<Stack spacing="xs">
|
<Stack spacing="xs">
|
||||||
{Object.entries(props.fields ?? {}).map(([fieldName, field]) => (
|
{Object.entries(props.fields ?? {}).map(([fieldName, field]) => (
|
||||||
<ApiFormField
|
<ApiFormField
|
||||||
@ -400,6 +421,7 @@ export function ApiForm({ id, props }: { id: string; props: ApiFormProps }) {
|
|||||||
/>
|
/>
|
||||||
))}
|
))}
|
||||||
</Stack>
|
</Stack>
|
||||||
|
</FormProvider>
|
||||||
{props.postFormContent}
|
{props.postFormContent}
|
||||||
</Stack>
|
</Stack>
|
||||||
<Divider />
|
<Divider />
|
||||||
|
@ -7,7 +7,6 @@ import {
|
|||||||
Switch,
|
Switch,
|
||||||
TextInput
|
TextInput
|
||||||
} from '@mantine/core';
|
} from '@mantine/core';
|
||||||
import { DateInput } from '@mantine/dates';
|
|
||||||
import { UseFormReturnType } from '@mantine/form';
|
import { UseFormReturnType } from '@mantine/form';
|
||||||
import { useId } from '@mantine/hooks';
|
import { useId } from '@mantine/hooks';
|
||||||
import { IconX } from '@tabler/icons-react';
|
import { IconX } from '@tabler/icons-react';
|
||||||
@ -17,11 +16,17 @@ import { Control, FieldValues, useController } from 'react-hook-form';
|
|||||||
|
|
||||||
import { ModelType } from '../../../enums/ModelType';
|
import { ModelType } from '../../../enums/ModelType';
|
||||||
import { ChoiceField } from './ChoiceField';
|
import { ChoiceField } from './ChoiceField';
|
||||||
|
import DateField from './DateField';
|
||||||
import { NestedObjectField } from './NestedObjectField';
|
import { NestedObjectField } from './NestedObjectField';
|
||||||
import { RelatedModelField } from './RelatedModelField';
|
import { RelatedModelField } from './RelatedModelField';
|
||||||
|
|
||||||
export type ApiFormData = UseFormReturnType<Record<string, unknown>>;
|
export type ApiFormData = UseFormReturnType<Record<string, unknown>>;
|
||||||
|
|
||||||
|
export type ApiFormAdjustFilterType = {
|
||||||
|
filters: any;
|
||||||
|
data: FieldValues;
|
||||||
|
};
|
||||||
|
|
||||||
/** Definition of the ApiForm field component.
|
/** Definition of the ApiForm field component.
|
||||||
* - The 'name' attribute *must* be provided
|
* - The 'name' attribute *must* be provided
|
||||||
* - All other attributes are optional, and may be provided by the API
|
* - All other attributes are optional, and may be provided by the API
|
||||||
@ -80,7 +85,7 @@ export type ApiFormFieldType = {
|
|||||||
preFieldContent?: JSX.Element;
|
preFieldContent?: JSX.Element;
|
||||||
postFieldContent?: JSX.Element;
|
postFieldContent?: JSX.Element;
|
||||||
onValueChange?: (value: any) => void;
|
onValueChange?: (value: any) => void;
|
||||||
adjustFilters?: (filters: any) => any;
|
adjustFilters?: (value: ApiFormAdjustFilterType) => any;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -207,20 +212,7 @@ export function ApiFormField({
|
|||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
case 'date':
|
case 'date':
|
||||||
return (
|
return <DateField controller={controller} definition={definition} />;
|
||||||
<DateInput
|
|
||||||
{...reducedDefinition}
|
|
||||||
ref={ref}
|
|
||||||
id={fieldId}
|
|
||||||
radius="sm"
|
|
||||||
type={undefined}
|
|
||||||
error={error?.message}
|
|
||||||
value={value}
|
|
||||||
clearable={!definition.required}
|
|
||||||
onChange={(value) => onChange(value)}
|
|
||||||
valueFormat="YYYY-MM-DD"
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
case 'integer':
|
case 'integer':
|
||||||
case 'decimal':
|
case 'decimal':
|
||||||
case 'float':
|
case 'float':
|
||||||
|
@ -58,6 +58,10 @@ export function ChoiceField({
|
|||||||
onChange={onChange}
|
onChange={onChange}
|
||||||
data={choices}
|
data={choices}
|
||||||
value={field.value}
|
value={field.value}
|
||||||
|
label={definition.label}
|
||||||
|
description={definition.description}
|
||||||
|
placeholder={definition.placeholder}
|
||||||
|
icon={definition.icon}
|
||||||
withinPortal={true}
|
withinPortal={true}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
|
60
src/frontend/src/components/forms/fields/DateField.tsx
Normal file
60
src/frontend/src/components/forms/fields/DateField.tsx
Normal file
@ -0,0 +1,60 @@
|
|||||||
|
import { DateInput } from '@mantine/dates';
|
||||||
|
import { useCallback, useId, useMemo } from 'react';
|
||||||
|
import { FieldValues, UseControllerReturn } from 'react-hook-form';
|
||||||
|
|
||||||
|
import { ApiFormFieldType } from './ApiFormField';
|
||||||
|
|
||||||
|
export default function DateField({
|
||||||
|
controller,
|
||||||
|
definition
|
||||||
|
}: {
|
||||||
|
controller: UseControllerReturn<FieldValues, any>;
|
||||||
|
definition: ApiFormFieldType;
|
||||||
|
}) {
|
||||||
|
const fieldId = useId();
|
||||||
|
|
||||||
|
const {
|
||||||
|
field,
|
||||||
|
fieldState: { error }
|
||||||
|
} = controller;
|
||||||
|
|
||||||
|
const onChange = useCallback(
|
||||||
|
(value: any) => {
|
||||||
|
// Convert the returned date object to a string
|
||||||
|
if (value) {
|
||||||
|
value = value.toString();
|
||||||
|
let date = new Date(value);
|
||||||
|
value = date.toISOString().split('T')[0];
|
||||||
|
}
|
||||||
|
|
||||||
|
field.onChange(value);
|
||||||
|
definition.onValueChange?.(value);
|
||||||
|
},
|
||||||
|
[field.onChange, definition]
|
||||||
|
);
|
||||||
|
|
||||||
|
const dateValue = useMemo(() => {
|
||||||
|
if (field.value) {
|
||||||
|
return new Date(field.value);
|
||||||
|
} else {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
}, [field.value]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<DateInput
|
||||||
|
id={fieldId}
|
||||||
|
radius="sm"
|
||||||
|
type={undefined}
|
||||||
|
error={error?.message}
|
||||||
|
value={dateValue}
|
||||||
|
clearable={!definition.required}
|
||||||
|
onChange={onChange}
|
||||||
|
valueFormat="YYYY-MM-DD"
|
||||||
|
label={definition.label}
|
||||||
|
description={definition.description}
|
||||||
|
placeholder={definition.placeholder}
|
||||||
|
icon={definition.icon}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
@ -4,7 +4,11 @@ import { useDebouncedValue } from '@mantine/hooks';
|
|||||||
import { useId } from '@mantine/hooks';
|
import { useId } from '@mantine/hooks';
|
||||||
import { useQuery } from '@tanstack/react-query';
|
import { useQuery } from '@tanstack/react-query';
|
||||||
import { useCallback, useEffect, useMemo, useRef, useState } from 'react';
|
import { useCallback, useEffect, useMemo, useRef, useState } from 'react';
|
||||||
import { FieldValues, UseControllerReturn } from 'react-hook-form';
|
import {
|
||||||
|
FieldValues,
|
||||||
|
UseControllerReturn,
|
||||||
|
useFormContext
|
||||||
|
} from 'react-hook-form';
|
||||||
import Select from 'react-select';
|
import Select from 'react-select';
|
||||||
|
|
||||||
import { api } from '../../../App';
|
import { api } from '../../../App';
|
||||||
@ -32,6 +36,8 @@ export function RelatedModelField({
|
|||||||
fieldState: { error }
|
fieldState: { error }
|
||||||
} = controller;
|
} = controller;
|
||||||
|
|
||||||
|
const form = useFormContext();
|
||||||
|
|
||||||
// Keep track of the primary key value for this field
|
// Keep track of the primary key value for this field
|
||||||
const [pk, setPk] = useState<number | null>(null);
|
const [pk, setPk] = useState<number | null>(null);
|
||||||
|
|
||||||
@ -40,6 +46,8 @@ export function RelatedModelField({
|
|||||||
const [data, setData] = useState<any[]>([]);
|
const [data, setData] = useState<any[]>([]);
|
||||||
const dataRef = useRef<any[]>([]);
|
const dataRef = useRef<any[]>([]);
|
||||||
|
|
||||||
|
const [isOpen, setIsOpen] = useState<boolean>(false);
|
||||||
|
|
||||||
// If an initial value is provided, load from the API
|
// If an initial value is provided, load from the API
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
// If the value is unchanged, do nothing
|
// If the value is unchanged, do nothing
|
||||||
@ -71,28 +79,49 @@ export function RelatedModelField({
|
|||||||
const [value, setValue] = useState<string>('');
|
const [value, setValue] = useState<string>('');
|
||||||
const [searchText, cancelSearchText] = useDebouncedValue(value, 250);
|
const [searchText, cancelSearchText] = useDebouncedValue(value, 250);
|
||||||
|
|
||||||
|
const [filters, setFilters] = useState<any>({});
|
||||||
|
|
||||||
|
const resetSearch = useCallback(() => {
|
||||||
|
setOffset(0);
|
||||||
|
setData([]);
|
||||||
|
dataRef.current = [];
|
||||||
|
}, []);
|
||||||
|
|
||||||
// reset current data on search value change
|
// reset current data on search value change
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
dataRef.current = [];
|
resetSearch();
|
||||||
setData([]);
|
}, [searchText, filters]);
|
||||||
}, [searchText]);
|
|
||||||
|
|
||||||
const selectQuery = useQuery({
|
const selectQuery = useQuery({
|
||||||
enabled: !definition.disabled && !!definition.api_url && !definition.hidden,
|
enabled:
|
||||||
|
isOpen &&
|
||||||
|
!definition.disabled &&
|
||||||
|
!!definition.api_url &&
|
||||||
|
!definition.hidden,
|
||||||
queryKey: [`related-field-${fieldName}`, fieldId, offset, searchText],
|
queryKey: [`related-field-${fieldName}`, fieldId, offset, searchText],
|
||||||
queryFn: async () => {
|
queryFn: async () => {
|
||||||
if (!definition.api_url) {
|
if (!definition.api_url) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
let filters = definition.filters ?? {};
|
let _filters = definition.filters ?? {};
|
||||||
|
|
||||||
if (definition.adjustFilters) {
|
if (definition.adjustFilters) {
|
||||||
filters = definition.adjustFilters(filters);
|
_filters =
|
||||||
|
definition.adjustFilters({
|
||||||
|
filters: _filters,
|
||||||
|
data: form.getValues()
|
||||||
|
}) ?? _filters;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If the filters have changed, clear the data
|
||||||
|
if (_filters != filters) {
|
||||||
|
resetSearch();
|
||||||
|
setFilters(_filters);
|
||||||
}
|
}
|
||||||
|
|
||||||
let params = {
|
let params = {
|
||||||
...filters,
|
..._filters,
|
||||||
search: searchText,
|
search: searchText,
|
||||||
offset: offset,
|
offset: offset,
|
||||||
limit: limit
|
limit: limit
|
||||||
@ -189,16 +218,19 @@ export function RelatedModelField({
|
|||||||
filterOption={null}
|
filterOption={null}
|
||||||
onInputChange={(value: any) => {
|
onInputChange={(value: any) => {
|
||||||
setValue(value);
|
setValue(value);
|
||||||
setOffset(0);
|
resetSearch();
|
||||||
setData([]);
|
|
||||||
}}
|
}}
|
||||||
onChange={onChange}
|
onChange={onChange}
|
||||||
onMenuScrollToBottom={() => setOffset(offset + limit)}
|
onMenuScrollToBottom={() => setOffset(offset + limit)}
|
||||||
onMenuOpen={() => {
|
onMenuOpen={() => {
|
||||||
|
setIsOpen(true);
|
||||||
setValue('');
|
setValue('');
|
||||||
setOffset(0);
|
resetSearch();
|
||||||
selectQuery.refetch();
|
selectQuery.refetch();
|
||||||
}}
|
}}
|
||||||
|
onMenuClose={() => {
|
||||||
|
setIsOpen(false);
|
||||||
|
}}
|
||||||
isLoading={
|
isLoading={
|
||||||
selectQuery.isFetching ||
|
selectQuery.isFetching ||
|
||||||
selectQuery.isLoading ||
|
selectQuery.isLoading ||
|
||||||
|
@ -7,7 +7,6 @@ import { RenderInlineModel } from './Instance';
|
|||||||
*/
|
*/
|
||||||
export function RenderAddress({ instance }: { instance: any }): ReactNode {
|
export function RenderAddress({ instance }: { instance: any }): ReactNode {
|
||||||
let text = [
|
let text = [
|
||||||
instance.title,
|
|
||||||
instance.country,
|
instance.country,
|
||||||
instance.postal_code,
|
instance.postal_code,
|
||||||
instance.postal_city,
|
instance.postal_city,
|
||||||
@ -18,12 +17,7 @@ export function RenderAddress({ instance }: { instance: any }): ReactNode {
|
|||||||
.filter(Boolean)
|
.filter(Boolean)
|
||||||
.join(', ');
|
.join(', ');
|
||||||
|
|
||||||
return (
|
return <RenderInlineModel primary={instance.title} secondary={text} />;
|
||||||
<RenderInlineModel
|
|
||||||
primary={instance.description}
|
|
||||||
secondary={instance.address}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -5,9 +5,14 @@ import { useNavigate } from 'react-router-dom';
|
|||||||
import { renderDate } from '../../../defaults/formatters';
|
import { renderDate } from '../../../defaults/formatters';
|
||||||
import { ApiEndpoints } from '../../../enums/ApiEndpoints';
|
import { ApiEndpoints } from '../../../enums/ApiEndpoints';
|
||||||
import { ModelType } from '../../../enums/ModelType';
|
import { ModelType } from '../../../enums/ModelType';
|
||||||
|
import { UserRoles } from '../../../enums/Roles';
|
||||||
|
import { buildOrderFields } from '../../../forms/BuildForms';
|
||||||
import { getDetailUrl } from '../../../functions/urls';
|
import { getDetailUrl } from '../../../functions/urls';
|
||||||
|
import { useCreateApiFormModal } from '../../../hooks/UseForm';
|
||||||
import { useTable } from '../../../hooks/UseTable';
|
import { useTable } from '../../../hooks/UseTable';
|
||||||
import { apiUrl } from '../../../states/ApiState';
|
import { apiUrl } from '../../../states/ApiState';
|
||||||
|
import { useUserState } from '../../../states/UserState';
|
||||||
|
import { AddItemButton } from '../../buttons/AddItemButton';
|
||||||
import { PartHoverCard } from '../../images/Thumbnail';
|
import { PartHoverCard } from '../../images/Thumbnail';
|
||||||
import { ProgressBar } from '../../items/ProgressBar';
|
import { ProgressBar } from '../../items/ProgressBar';
|
||||||
import { RenderUser } from '../../render/User';
|
import { RenderUser } from '../../render/User';
|
||||||
@ -88,7 +93,15 @@ function buildOrderTableColumns(): TableColumn[] {
|
|||||||
/*
|
/*
|
||||||
* Construct a table of build orders, according to the provided parameters
|
* Construct a table of build orders, according to the provided parameters
|
||||||
*/
|
*/
|
||||||
export function BuildOrderTable({ params = {} }: { params?: any }) {
|
export function BuildOrderTable({
|
||||||
|
partId,
|
||||||
|
parentBuildId,
|
||||||
|
salesOrderId
|
||||||
|
}: {
|
||||||
|
partId?: number;
|
||||||
|
parentBuildId?: number;
|
||||||
|
salesOrderId?: number;
|
||||||
|
}) {
|
||||||
const tableColumns = useMemo(() => buildOrderTableColumns(), []);
|
const tableColumns = useMemo(() => buildOrderTableColumns(), []);
|
||||||
|
|
||||||
const tableFilters: TableFilter[] = useMemo(() => {
|
const tableFilters: TableFilter[] = useMemo(() => {
|
||||||
@ -130,10 +143,39 @@ export function BuildOrderTable({ params = {} }: { params?: any }) {
|
|||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
|
const user = useUserState();
|
||||||
|
|
||||||
const table = useTable('buildorder');
|
const table = useTable('buildorder');
|
||||||
|
|
||||||
|
const newBuild = useCreateApiFormModal({
|
||||||
|
url: ApiEndpoints.build_order_list,
|
||||||
|
title: t`Add Build Order`,
|
||||||
|
fields: buildOrderFields(),
|
||||||
|
initialData: {
|
||||||
|
part: partId,
|
||||||
|
sales_order: salesOrderId,
|
||||||
|
parent: parentBuildId
|
||||||
|
},
|
||||||
|
onFormSuccess: (data: any) => {
|
||||||
|
if (data.pk) {
|
||||||
|
navigate(getDetailUrl(ModelType.build, data.pk));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const tableActions = useMemo(() => {
|
||||||
|
return [
|
||||||
|
<AddItemButton
|
||||||
|
hidden={!user.hasAddRole(UserRoles.build)}
|
||||||
|
tooltip={t`Add Build Order`}
|
||||||
|
onClick={() => newBuild.open()}
|
||||||
|
/>
|
||||||
|
];
|
||||||
|
}, [user]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
<>
|
||||||
|
{newBuild.modal}
|
||||||
<InvenTreeTable
|
<InvenTreeTable
|
||||||
url={apiUrl(ApiEndpoints.build_order_list)}
|
url={apiUrl(ApiEndpoints.build_order_list)}
|
||||||
tableState={table}
|
tableState={table}
|
||||||
@ -141,12 +183,16 @@ export function BuildOrderTable({ params = {} }: { params?: any }) {
|
|||||||
props={{
|
props={{
|
||||||
enableDownload: true,
|
enableDownload: true,
|
||||||
params: {
|
params: {
|
||||||
...params,
|
part: partId,
|
||||||
|
sales_order: salesOrderId,
|
||||||
|
parent: parentBuildId,
|
||||||
part_detail: true
|
part_detail: true
|
||||||
},
|
},
|
||||||
|
tableActions: tableActions,
|
||||||
tableFilters: tableFilters,
|
tableFilters: tableFilters,
|
||||||
onRowClick: (row) => navigate(getDetailUrl(ModelType.build, row.pk))
|
onRowClick: (row) => navigate(getDetailUrl(ModelType.build, row.pk))
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -1,18 +1,18 @@
|
|||||||
import { t } from '@lingui/macro';
|
import { t } from '@lingui/macro';
|
||||||
import { useCallback, useMemo } from 'react';
|
import { useCallback, useMemo, useState } from 'react';
|
||||||
|
|
||||||
import { ApiEndpoints } from '../../../enums/ApiEndpoints';
|
import { ApiEndpoints } from '../../../enums/ApiEndpoints';
|
||||||
import { UserRoles } from '../../../enums/Roles';
|
import { UserRoles } from '../../../enums/Roles';
|
||||||
import { addressFields } from '../../../forms/CompanyForms';
|
|
||||||
import {
|
import {
|
||||||
openCreateApiForm,
|
useCreateApiFormModal,
|
||||||
openDeleteApiForm,
|
useDeleteApiFormModal,
|
||||||
openEditApiForm
|
useEditApiFormModal
|
||||||
} from '../../../functions/forms';
|
} from '../../../hooks/UseForm';
|
||||||
import { useTable } from '../../../hooks/UseTable';
|
import { useTable } from '../../../hooks/UseTable';
|
||||||
import { apiUrl } from '../../../states/ApiState';
|
import { apiUrl } from '../../../states/ApiState';
|
||||||
import { useUserState } from '../../../states/UserState';
|
import { useUserState } from '../../../states/UserState';
|
||||||
import { AddItemButton } from '../../buttons/AddItemButton';
|
import { AddItemButton } from '../../buttons/AddItemButton';
|
||||||
|
import { ApiFormFieldSet } from '../../forms/fields/ApiFormField';
|
||||||
import { YesNoButton } from '../../items/YesNoButton';
|
import { YesNoButton } from '../../items/YesNoButton';
|
||||||
import { TableColumn } from '../Column';
|
import { TableColumn } from '../Column';
|
||||||
import { InvenTreeTable } from '../InvenTreeTable';
|
import { InvenTreeTable } from '../InvenTreeTable';
|
||||||
@ -108,6 +108,54 @@ export function AddressTable({
|
|||||||
];
|
];
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
const addressFields: ApiFormFieldSet = useMemo(() => {
|
||||||
|
return {
|
||||||
|
company: {},
|
||||||
|
title: {},
|
||||||
|
primary: {},
|
||||||
|
line1: {},
|
||||||
|
line2: {},
|
||||||
|
postal_code: {},
|
||||||
|
postal_city: {},
|
||||||
|
province: {},
|
||||||
|
country: {},
|
||||||
|
shipping_notes: {},
|
||||||
|
internal_shipping_notes: {},
|
||||||
|
link: {}
|
||||||
|
};
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const newAddress = useCreateApiFormModal({
|
||||||
|
url: ApiEndpoints.address_list,
|
||||||
|
title: t`Add Address`,
|
||||||
|
fields: addressFields,
|
||||||
|
initialData: {
|
||||||
|
company: companyId
|
||||||
|
},
|
||||||
|
successMessage: t`Address created`,
|
||||||
|
onFormSuccess: table.refreshTable
|
||||||
|
});
|
||||||
|
|
||||||
|
const [selectedAddress, setSelectedAddress] = useState<number | undefined>(
|
||||||
|
undefined
|
||||||
|
);
|
||||||
|
|
||||||
|
const editAddress = useEditApiFormModal({
|
||||||
|
url: ApiEndpoints.address_list,
|
||||||
|
pk: selectedAddress,
|
||||||
|
title: t`Edit Address`,
|
||||||
|
fields: addressFields,
|
||||||
|
onFormSuccess: table.refreshTable
|
||||||
|
});
|
||||||
|
|
||||||
|
const deleteAddress = useDeleteApiFormModal({
|
||||||
|
url: ApiEndpoints.address_list,
|
||||||
|
pk: selectedAddress,
|
||||||
|
title: t`Delete Address`,
|
||||||
|
onFormSuccess: table.refreshTable,
|
||||||
|
preFormWarning: t`Are you sure you want to delete this address?`
|
||||||
|
});
|
||||||
|
|
||||||
const rowActions = useCallback(
|
const rowActions = useCallback(
|
||||||
(record: any) => {
|
(record: any) => {
|
||||||
let can_edit =
|
let can_edit =
|
||||||
@ -122,27 +170,15 @@ export function AddressTable({
|
|||||||
RowEditAction({
|
RowEditAction({
|
||||||
hidden: !can_edit,
|
hidden: !can_edit,
|
||||||
onClick: () => {
|
onClick: () => {
|
||||||
openEditApiForm({
|
setSelectedAddress(record.pk);
|
||||||
url: ApiEndpoints.address_list,
|
editAddress.open();
|
||||||
pk: record.pk,
|
|
||||||
title: t`Edit Address`,
|
|
||||||
fields: addressFields(),
|
|
||||||
successMessage: t`Address updated`,
|
|
||||||
onFormSuccess: table.refreshTable
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
}),
|
}),
|
||||||
RowDeleteAction({
|
RowDeleteAction({
|
||||||
hidden: !can_delete,
|
hidden: !can_delete,
|
||||||
onClick: () => {
|
onClick: () => {
|
||||||
openDeleteApiForm({
|
setSelectedAddress(record.pk);
|
||||||
url: ApiEndpoints.address_list,
|
deleteAddress.open();
|
||||||
pk: record.pk,
|
|
||||||
title: t`Delete Address`,
|
|
||||||
successMessage: t`Address deleted`,
|
|
||||||
onFormSuccess: table.refreshTable,
|
|
||||||
preFormWarning: t`Are you sure you want to delete this address?`
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
];
|
];
|
||||||
@ -150,20 +186,6 @@ export function AddressTable({
|
|||||||
[user]
|
[user]
|
||||||
);
|
);
|
||||||
|
|
||||||
const addAddress = useCallback(() => {
|
|
||||||
let fields = addressFields();
|
|
||||||
|
|
||||||
fields['company'].value = companyId;
|
|
||||||
|
|
||||||
openCreateApiForm({
|
|
||||||
url: ApiEndpoints.address_list,
|
|
||||||
title: t`Add Address`,
|
|
||||||
fields: fields,
|
|
||||||
successMessage: t`Address created`,
|
|
||||||
onFormSuccess: table.refreshTable
|
|
||||||
});
|
|
||||||
}, [companyId]);
|
|
||||||
|
|
||||||
const tableActions = useMemo(() => {
|
const tableActions = useMemo(() => {
|
||||||
let can_add =
|
let can_add =
|
||||||
user.hasChangeRole(UserRoles.purchase_order) ||
|
user.hasChangeRole(UserRoles.purchase_order) ||
|
||||||
@ -172,13 +194,17 @@ export function AddressTable({
|
|||||||
return [
|
return [
|
||||||
<AddItemButton
|
<AddItemButton
|
||||||
tooltip={t`Add Address`}
|
tooltip={t`Add Address`}
|
||||||
onClick={addAddress}
|
onClick={() => newAddress.open()}
|
||||||
disabled={!can_add}
|
disabled={!can_add}
|
||||||
/>
|
/>
|
||||||
];
|
];
|
||||||
}, [user]);
|
}, [user]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
<>
|
||||||
|
{newAddress.modal}
|
||||||
|
{editAddress.modal}
|
||||||
|
{deleteAddress.modal}
|
||||||
<InvenTreeTable
|
<InvenTreeTable
|
||||||
url={apiUrl(ApiEndpoints.address_list)}
|
url={apiUrl(ApiEndpoints.address_list)}
|
||||||
tableState={table}
|
tableState={table}
|
||||||
@ -192,5 +218,6 @@ export function AddressTable({
|
|||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -4,8 +4,13 @@ import { useMemo } from 'react';
|
|||||||
import { useNavigate } from 'react-router-dom';
|
import { useNavigate } from 'react-router-dom';
|
||||||
|
|
||||||
import { ApiEndpoints } from '../../../enums/ApiEndpoints';
|
import { ApiEndpoints } from '../../../enums/ApiEndpoints';
|
||||||
|
import { UserRoles } from '../../../enums/Roles';
|
||||||
|
import { companyFields } from '../../../forms/CompanyForms';
|
||||||
|
import { useCreateApiFormModal } from '../../../hooks/UseForm';
|
||||||
import { useTable } from '../../../hooks/UseTable';
|
import { useTable } from '../../../hooks/UseTable';
|
||||||
import { apiUrl } from '../../../states/ApiState';
|
import { apiUrl } from '../../../states/ApiState';
|
||||||
|
import { useUserState } from '../../../states/UserState';
|
||||||
|
import { AddItemButton } from '../../buttons/AddItemButton';
|
||||||
import { Thumbnail } from '../../images/Thumbnail';
|
import { Thumbnail } from '../../images/Thumbnail';
|
||||||
import { DescriptionColumn } from '../ColumnRenderers';
|
import { DescriptionColumn } from '../ColumnRenderers';
|
||||||
import { InvenTreeTable } from '../InvenTreeTable';
|
import { InvenTreeTable } from '../InvenTreeTable';
|
||||||
@ -24,6 +29,7 @@ export function CompanyTable({
|
|||||||
const table = useTable('company');
|
const table = useTable('company');
|
||||||
|
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
|
const user = useUserState();
|
||||||
|
|
||||||
const columns = useMemo(() => {
|
const columns = useMemo(() => {
|
||||||
return [
|
return [
|
||||||
@ -53,7 +59,39 @@ export function CompanyTable({
|
|||||||
];
|
];
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
const newCompany = useCreateApiFormModal({
|
||||||
|
url: ApiEndpoints.company_list,
|
||||||
|
title: t`New Company`,
|
||||||
|
fields: companyFields(),
|
||||||
|
initialData: params,
|
||||||
|
onFormSuccess: (response) => {
|
||||||
|
console.log('onFormSuccess:', response);
|
||||||
|
if (response.pk) {
|
||||||
|
let base = path ?? 'company';
|
||||||
|
navigate(`/${base}/${response.pk}`);
|
||||||
|
} else {
|
||||||
|
table.refreshTable();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const tableActions = useMemo(() => {
|
||||||
|
const can_add =
|
||||||
|
user.hasAddRole(UserRoles.purchase_order) ||
|
||||||
|
user.hasAddRole(UserRoles.sales_order);
|
||||||
|
|
||||||
|
return [
|
||||||
|
<AddItemButton
|
||||||
|
tooltip={t`Add Company`}
|
||||||
|
onClick={() => newCompany.open()}
|
||||||
|
hidden={!can_add}
|
||||||
|
/>
|
||||||
|
];
|
||||||
|
}, [user]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
<>
|
||||||
|
{newCompany.modal}
|
||||||
<InvenTreeTable
|
<InvenTreeTable
|
||||||
url={apiUrl(ApiEndpoints.company_list)}
|
url={apiUrl(ApiEndpoints.company_list)}
|
||||||
tableState={table}
|
tableState={table}
|
||||||
@ -62,6 +100,7 @@ export function CompanyTable({
|
|||||||
params: {
|
params: {
|
||||||
...params
|
...params
|
||||||
},
|
},
|
||||||
|
tableActions: tableActions,
|
||||||
onRowClick: (row: any) => {
|
onRowClick: (row: any) => {
|
||||||
if (row.pk) {
|
if (row.pk) {
|
||||||
let base = path ?? 'company';
|
let base = path ?? 'company';
|
||||||
@ -70,5 +109,6 @@ export function CompanyTable({
|
|||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -1,18 +1,18 @@
|
|||||||
import { t } from '@lingui/macro';
|
import { t } from '@lingui/macro';
|
||||||
import { useCallback, useMemo } from 'react';
|
import { useCallback, useMemo, useState } from 'react';
|
||||||
|
|
||||||
import { ApiEndpoints } from '../../../enums/ApiEndpoints';
|
import { ApiEndpoints } from '../../../enums/ApiEndpoints';
|
||||||
import { UserRoles } from '../../../enums/Roles';
|
import { UserRoles } from '../../../enums/Roles';
|
||||||
import { contactFields } from '../../../forms/CompanyForms';
|
|
||||||
import {
|
import {
|
||||||
openCreateApiForm,
|
useCreateApiFormModal,
|
||||||
openDeleteApiForm,
|
useDeleteApiFormModal,
|
||||||
openEditApiForm
|
useEditApiFormModal
|
||||||
} from '../../../functions/forms';
|
} from '../../../hooks/UseForm';
|
||||||
import { useTable } from '../../../hooks/UseTable';
|
import { useTable } from '../../../hooks/UseTable';
|
||||||
import { apiUrl } from '../../../states/ApiState';
|
import { apiUrl } from '../../../states/ApiState';
|
||||||
import { useUserState } from '../../../states/UserState';
|
import { useUserState } from '../../../states/UserState';
|
||||||
import { AddItemButton } from '../../buttons/AddItemButton';
|
import { AddItemButton } from '../../buttons/AddItemButton';
|
||||||
|
import { ApiFormFieldSet } from '../../forms/fields/ApiFormField';
|
||||||
import { TableColumn } from '../Column';
|
import { TableColumn } from '../Column';
|
||||||
import { InvenTreeTable } from '../InvenTreeTable';
|
import { InvenTreeTable } from '../InvenTreeTable';
|
||||||
import { RowDeleteAction, RowEditAction } from '../RowActions';
|
import { RowDeleteAction, RowEditAction } from '../RowActions';
|
||||||
@ -57,6 +57,45 @@ export function ContactTable({
|
|||||||
];
|
];
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
const contactFields: ApiFormFieldSet = useMemo(() => {
|
||||||
|
return {
|
||||||
|
company: {},
|
||||||
|
name: {},
|
||||||
|
phone: {},
|
||||||
|
email: {},
|
||||||
|
role: {}
|
||||||
|
};
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const [selectedContact, setSelectedContact] = useState<number | undefined>(
|
||||||
|
undefined
|
||||||
|
);
|
||||||
|
|
||||||
|
const editContact = useEditApiFormModal({
|
||||||
|
url: ApiEndpoints.contact_list,
|
||||||
|
pk: selectedContact,
|
||||||
|
title: t`Edit Contact`,
|
||||||
|
fields: contactFields,
|
||||||
|
onFormSuccess: table.refreshTable
|
||||||
|
});
|
||||||
|
|
||||||
|
const newContact = useCreateApiFormModal({
|
||||||
|
url: ApiEndpoints.contact_list,
|
||||||
|
title: t`Add Contact`,
|
||||||
|
initialData: {
|
||||||
|
company: companyId
|
||||||
|
},
|
||||||
|
fields: contactFields,
|
||||||
|
onFormSuccess: table.refreshTable
|
||||||
|
});
|
||||||
|
|
||||||
|
const deleteContact = useDeleteApiFormModal({
|
||||||
|
url: ApiEndpoints.contact_list,
|
||||||
|
pk: selectedContact,
|
||||||
|
title: t`Delete Contact`,
|
||||||
|
onFormSuccess: table.refreshTable
|
||||||
|
});
|
||||||
|
|
||||||
const rowActions = useCallback(
|
const rowActions = useCallback(
|
||||||
(record: any) => {
|
(record: any) => {
|
||||||
let can_edit =
|
let can_edit =
|
||||||
@ -70,27 +109,15 @@ export function ContactTable({
|
|||||||
RowEditAction({
|
RowEditAction({
|
||||||
hidden: !can_edit,
|
hidden: !can_edit,
|
||||||
onClick: () => {
|
onClick: () => {
|
||||||
openEditApiForm({
|
setSelectedContact(record.pk);
|
||||||
url: ApiEndpoints.contact_list,
|
editContact.open();
|
||||||
pk: record.pk,
|
|
||||||
title: t`Edit Contact`,
|
|
||||||
fields: contactFields(),
|
|
||||||
successMessage: t`Contact updated`,
|
|
||||||
onFormSuccess: table.refreshTable
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
}),
|
}),
|
||||||
RowDeleteAction({
|
RowDeleteAction({
|
||||||
hidden: !can_delete,
|
hidden: !can_delete,
|
||||||
onClick: () => {
|
onClick: () => {
|
||||||
openDeleteApiForm({
|
setSelectedContact(record.pk);
|
||||||
url: ApiEndpoints.contact_list,
|
deleteContact.open();
|
||||||
pk: record.pk,
|
|
||||||
title: t`Delete Contact`,
|
|
||||||
successMessage: t`Contact deleted`,
|
|
||||||
onFormSuccess: table.refreshTable,
|
|
||||||
preFormWarning: t`Are you sure you want to delete this contact?`
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
];
|
];
|
||||||
@ -98,20 +125,6 @@ export function ContactTable({
|
|||||||
[user]
|
[user]
|
||||||
);
|
);
|
||||||
|
|
||||||
const addContact = useCallback(() => {
|
|
||||||
var fields = contactFields();
|
|
||||||
|
|
||||||
fields['company'].value = companyId;
|
|
||||||
|
|
||||||
openCreateApiForm({
|
|
||||||
url: ApiEndpoints.contact_list,
|
|
||||||
title: t`Create Contact`,
|
|
||||||
fields: fields,
|
|
||||||
successMessage: t`Contact created`,
|
|
||||||
onFormSuccess: table.refreshTable
|
|
||||||
});
|
|
||||||
}, [companyId]);
|
|
||||||
|
|
||||||
const tableActions = useMemo(() => {
|
const tableActions = useMemo(() => {
|
||||||
let can_add =
|
let can_add =
|
||||||
user.hasAddRole(UserRoles.purchase_order) ||
|
user.hasAddRole(UserRoles.purchase_order) ||
|
||||||
@ -120,13 +133,17 @@ export function ContactTable({
|
|||||||
return [
|
return [
|
||||||
<AddItemButton
|
<AddItemButton
|
||||||
tooltip={t`Add contact`}
|
tooltip={t`Add contact`}
|
||||||
onClick={addContact}
|
onClick={() => newContact.open()}
|
||||||
disabled={!can_add}
|
disabled={!can_add}
|
||||||
/>
|
/>
|
||||||
];
|
];
|
||||||
}, [user]);
|
}, [user]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
<>
|
||||||
|
{newContact.modal}
|
||||||
|
{editContact.modal}
|
||||||
|
{deleteContact.modal}
|
||||||
<InvenTreeTable
|
<InvenTreeTable
|
||||||
url={apiUrl(ApiEndpoints.contact_list)}
|
url={apiUrl(ApiEndpoints.contact_list)}
|
||||||
tableState={table}
|
tableState={table}
|
||||||
@ -140,5 +157,6 @@ export function ContactTable({
|
|||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -1,13 +1,16 @@
|
|||||||
import { t } from '@lingui/macro';
|
import { t } from '@lingui/macro';
|
||||||
import { useCallback, useMemo } from 'react';
|
import { useCallback, useMemo, useState } from 'react';
|
||||||
import { useNavigate } from 'react-router-dom';
|
import { useNavigate } from 'react-router-dom';
|
||||||
|
|
||||||
import { ApiEndpoints } from '../../../enums/ApiEndpoints';
|
import { ApiEndpoints } from '../../../enums/ApiEndpoints';
|
||||||
import { ModelType } from '../../../enums/ModelType';
|
import { ModelType } from '../../../enums/ModelType';
|
||||||
import { UserRoles } from '../../../enums/Roles';
|
import { UserRoles } from '../../../enums/Roles';
|
||||||
import { partCategoryFields } from '../../../forms/PartForms';
|
import { partCategoryFields } from '../../../forms/PartForms';
|
||||||
import { openCreateApiForm, openEditApiForm } from '../../../functions/forms';
|
|
||||||
import { getDetailUrl } from '../../../functions/urls';
|
import { getDetailUrl } from '../../../functions/urls';
|
||||||
|
import {
|
||||||
|
useCreateApiFormModal,
|
||||||
|
useEditApiFormModal
|
||||||
|
} from '../../../hooks/UseForm';
|
||||||
import { useTable } from '../../../hooks/UseTable';
|
import { useTable } from '../../../hooks/UseTable';
|
||||||
import { apiUrl } from '../../../states/ApiState';
|
import { apiUrl } from '../../../states/ApiState';
|
||||||
import { useUserState } from '../../../states/UserState';
|
import { useUserState } from '../../../states/UserState';
|
||||||
@ -73,26 +76,33 @@ export function PartCategoryTable({ parentId }: { parentId?: any }) {
|
|||||||
];
|
];
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const addCategory = useCallback(() => {
|
const newCategory = useCreateApiFormModal({
|
||||||
let fields = partCategoryFields({});
|
url: ApiEndpoints.category_list,
|
||||||
|
title: t`New Part Category`,
|
||||||
if (parentId) {
|
fields: partCategoryFields({}),
|
||||||
fields['parent'].value = parentId;
|
initialData: {
|
||||||
}
|
parent: parentId
|
||||||
|
},
|
||||||
openCreateApiForm({
|
|
||||||
url: apiUrl(ApiEndpoints.category_list),
|
|
||||||
title: t`Add Part Category`,
|
|
||||||
fields: fields,
|
|
||||||
onFormSuccess(data: any) {
|
onFormSuccess(data: any) {
|
||||||
if (data.pk) {
|
if (data.pk) {
|
||||||
navigate(`/part/category/${data.pk}`);
|
navigate(getDetailUrl(ModelType.partcategory, data.pk));
|
||||||
} else {
|
} else {
|
||||||
table.refreshTable();
|
table.refreshTable();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}, [parentId]);
|
|
||||||
|
const [selectedCategory, setSelectedCategory] = useState<number | undefined>(
|
||||||
|
undefined
|
||||||
|
);
|
||||||
|
|
||||||
|
const editCategory = useEditApiFormModal({
|
||||||
|
url: ApiEndpoints.category_list,
|
||||||
|
pk: selectedCategory,
|
||||||
|
title: t`Edit Part Category`,
|
||||||
|
fields: partCategoryFields({}),
|
||||||
|
onFormSuccess: table.refreshTable
|
||||||
|
});
|
||||||
|
|
||||||
const tableActions = useMemo(() => {
|
const tableActions = useMemo(() => {
|
||||||
let can_add = user.hasAddRole(UserRoles.part_category);
|
let can_add = user.hasAddRole(UserRoles.part_category);
|
||||||
@ -100,7 +110,7 @@ export function PartCategoryTable({ parentId }: { parentId?: any }) {
|
|||||||
return [
|
return [
|
||||||
<AddItemButton
|
<AddItemButton
|
||||||
tooltip={t`Add Part Category`}
|
tooltip={t`Add Part Category`}
|
||||||
onClick={addCategory}
|
onClick={() => newCategory.open()}
|
||||||
disabled={!can_add}
|
disabled={!can_add}
|
||||||
/>
|
/>
|
||||||
];
|
];
|
||||||
@ -114,14 +124,8 @@ export function PartCategoryTable({ parentId }: { parentId?: any }) {
|
|||||||
RowEditAction({
|
RowEditAction({
|
||||||
hidden: !can_edit,
|
hidden: !can_edit,
|
||||||
onClick: () => {
|
onClick: () => {
|
||||||
openEditApiForm({
|
setSelectedCategory(record.pk);
|
||||||
url: ApiEndpoints.category_list,
|
editCategory.open();
|
||||||
pk: record.pk,
|
|
||||||
title: t`Edit Part Category`,
|
|
||||||
fields: partCategoryFields({}),
|
|
||||||
successMessage: t`Part category updated`,
|
|
||||||
onFormSuccess: table.refreshTable
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
];
|
];
|
||||||
@ -130,6 +134,9 @@ export function PartCategoryTable({ parentId }: { parentId?: any }) {
|
|||||||
);
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
<>
|
||||||
|
{newCategory.modal}
|
||||||
|
{editCategory.modal}
|
||||||
<InvenTreeTable
|
<InvenTreeTable
|
||||||
url={apiUrl(ApiEndpoints.category_list)}
|
url={apiUrl(ApiEndpoints.category_list)}
|
||||||
tableState={table}
|
tableState={table}
|
||||||
@ -142,9 +149,10 @@ export function PartCategoryTable({ parentId }: { parentId?: any }) {
|
|||||||
tableFilters: tableFilters,
|
tableFilters: tableFilters,
|
||||||
tableActions: tableActions,
|
tableActions: tableActions,
|
||||||
rowActions: rowActions,
|
rowActions: rowActions,
|
||||||
onRowClick: (record, index, event) =>
|
onRowClick: (record) =>
|
||||||
navigate(getDetailUrl(ModelType.partcategory, record.pk))
|
navigate(getDetailUrl(ModelType.partcategory, record.pk))
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -1,18 +1,19 @@
|
|||||||
import { t } from '@lingui/macro';
|
import { t } from '@lingui/macro';
|
||||||
import { Text } from '@mantine/core';
|
import { Text } from '@mantine/core';
|
||||||
import { useCallback, useMemo } from 'react';
|
import { useCallback, useMemo, useState } from 'react';
|
||||||
|
|
||||||
import { ApiEndpoints } from '../../../enums/ApiEndpoints';
|
import { ApiEndpoints } from '../../../enums/ApiEndpoints';
|
||||||
import { UserRoles } from '../../../enums/Roles';
|
import { UserRoles } from '../../../enums/Roles';
|
||||||
import {
|
import {
|
||||||
openCreateApiForm,
|
useCreateApiFormModal,
|
||||||
openDeleteApiForm,
|
useDeleteApiFormModal,
|
||||||
openEditApiForm
|
useEditApiFormModal
|
||||||
} from '../../../functions/forms';
|
} from '../../../hooks/UseForm';
|
||||||
import { useTable } from '../../../hooks/UseTable';
|
import { useTable } from '../../../hooks/UseTable';
|
||||||
import { apiUrl } from '../../../states/ApiState';
|
import { apiUrl } from '../../../states/ApiState';
|
||||||
import { useUserState } from '../../../states/UserState';
|
import { useUserState } from '../../../states/UserState';
|
||||||
import { AddItemButton } from '../../buttons/AddItemButton';
|
import { AddItemButton } from '../../buttons/AddItemButton';
|
||||||
|
import { ApiFormFieldSet } from '../../forms/fields/ApiFormField';
|
||||||
import { YesNoButton } from '../../items/YesNoButton';
|
import { YesNoButton } from '../../items/YesNoButton';
|
||||||
import { TableColumn } from '../Column';
|
import { TableColumn } from '../Column';
|
||||||
import { PartColumn } from '../ColumnRenderers';
|
import { PartColumn } from '../ColumnRenderers';
|
||||||
@ -36,6 +37,12 @@ export function PartParameterTable({ partId }: { partId: any }) {
|
|||||||
sortable: true,
|
sortable: true,
|
||||||
render: (record: any) => PartColumn(record?.part_detail)
|
render: (record: any) => PartColumn(record?.part_detail)
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
accessor: 'part_detail.IPN',
|
||||||
|
title: t`IPN`,
|
||||||
|
sortable: false,
|
||||||
|
switchable: true
|
||||||
|
},
|
||||||
{
|
{
|
||||||
accessor: 'name',
|
accessor: 'name',
|
||||||
title: t`Parameter`,
|
title: t`Parameter`,
|
||||||
@ -85,6 +92,43 @@ export function PartParameterTable({ partId }: { partId: any }) {
|
|||||||
];
|
];
|
||||||
}, [partId]);
|
}, [partId]);
|
||||||
|
|
||||||
|
const partParameterFields: ApiFormFieldSet = useMemo(() => {
|
||||||
|
return {
|
||||||
|
part: {},
|
||||||
|
template: {},
|
||||||
|
data: {}
|
||||||
|
};
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const newParameter = useCreateApiFormModal({
|
||||||
|
url: ApiEndpoints.part_parameter_list,
|
||||||
|
title: t`New Part Parameter`,
|
||||||
|
fields: partParameterFields,
|
||||||
|
initialData: {
|
||||||
|
part: partId
|
||||||
|
},
|
||||||
|
onFormSuccess: table.refreshTable
|
||||||
|
});
|
||||||
|
|
||||||
|
const [selectedParameter, setSelectedParameter] = useState<
|
||||||
|
number | undefined
|
||||||
|
>(undefined);
|
||||||
|
|
||||||
|
const editParameter = useEditApiFormModal({
|
||||||
|
url: ApiEndpoints.part_parameter_list,
|
||||||
|
pk: selectedParameter,
|
||||||
|
title: t`Edit Part Parameter`,
|
||||||
|
fields: partParameterFields,
|
||||||
|
onFormSuccess: table.refreshTable
|
||||||
|
});
|
||||||
|
|
||||||
|
const deleteParameter = useDeleteApiFormModal({
|
||||||
|
url: ApiEndpoints.part_parameter_list,
|
||||||
|
pk: selectedParameter,
|
||||||
|
title: t`Delete Part Parameter`,
|
||||||
|
onFormSuccess: table.refreshTable
|
||||||
|
});
|
||||||
|
|
||||||
// Callback for row actions
|
// Callback for row actions
|
||||||
const rowActions = useCallback(
|
const rowActions = useCallback(
|
||||||
(record: any) => {
|
(record: any) => {
|
||||||
@ -93,87 +137,44 @@ export function PartParameterTable({ partId }: { partId: any }) {
|
|||||||
return [];
|
return [];
|
||||||
}
|
}
|
||||||
|
|
||||||
let actions = [];
|
return [
|
||||||
|
|
||||||
actions.push(
|
|
||||||
RowEditAction({
|
RowEditAction({
|
||||||
tooltip: t`Edit Part Parameter`,
|
tooltip: t`Edit Part Parameter`,
|
||||||
hidden: !user.hasChangeRole(UserRoles.part),
|
hidden: !user.hasChangeRole(UserRoles.part),
|
||||||
onClick: () => {
|
onClick: () => {
|
||||||
openEditApiForm({
|
setSelectedParameter(record.pk);
|
||||||
url: ApiEndpoints.part_parameter_list,
|
editParameter.open();
|
||||||
pk: record.pk,
|
|
||||||
title: t`Edit Part Parameter`,
|
|
||||||
fields: {
|
|
||||||
part: {
|
|
||||||
hidden: true
|
|
||||||
},
|
|
||||||
template: {},
|
|
||||||
data: {}
|
|
||||||
},
|
|
||||||
successMessage: t`Part parameter updated`,
|
|
||||||
onFormSuccess: table.refreshTable
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
})
|
}),
|
||||||
);
|
|
||||||
|
|
||||||
actions.push(
|
|
||||||
RowDeleteAction({
|
RowDeleteAction({
|
||||||
tooltip: t`Delete Part Parameter`,
|
tooltip: t`Delete Part Parameter`,
|
||||||
hidden: !user.hasDeleteRole(UserRoles.part),
|
hidden: !user.hasDeleteRole(UserRoles.part),
|
||||||
onClick: () => {
|
onClick: () => {
|
||||||
openDeleteApiForm({
|
setSelectedParameter(record.pk);
|
||||||
url: ApiEndpoints.part_parameter_list,
|
deleteParameter.open();
|
||||||
pk: record.pk,
|
|
||||||
title: t`Delete Part Parameter`,
|
|
||||||
successMessage: t`Part parameter deleted`,
|
|
||||||
onFormSuccess: table.refreshTable,
|
|
||||||
preFormWarning: t`Are you sure you want to remove this parameter?`
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
);
|
];
|
||||||
|
|
||||||
return actions;
|
|
||||||
},
|
},
|
||||||
[partId, user]
|
[partId, user]
|
||||||
);
|
);
|
||||||
|
|
||||||
const addParameter = useCallback(() => {
|
|
||||||
if (!partId) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
openCreateApiForm({
|
|
||||||
url: ApiEndpoints.part_parameter_list,
|
|
||||||
title: t`Add Part Parameter`,
|
|
||||||
fields: {
|
|
||||||
part: {
|
|
||||||
hidden: true,
|
|
||||||
value: partId
|
|
||||||
},
|
|
||||||
template: {},
|
|
||||||
data: {}
|
|
||||||
},
|
|
||||||
successMessage: t`Part parameter added`,
|
|
||||||
onFormSuccess: table.refreshTable
|
|
||||||
});
|
|
||||||
}, [partId]);
|
|
||||||
|
|
||||||
// Custom table actions
|
// Custom table actions
|
||||||
const tableActions = useMemo(() => {
|
const tableActions = useMemo(() => {
|
||||||
let actions = [];
|
return [
|
||||||
|
<AddItemButton
|
||||||
// TODO: Hide if user does not have permission to edit parts
|
hidden={!user.hasAddRole(UserRoles.part)}
|
||||||
actions.push(
|
tooltip={t`Add parameter`}
|
||||||
<AddItemButton tooltip={t`Add parameter`} onClick={addParameter} />
|
onClick={() => newParameter.open()}
|
||||||
);
|
/>
|
||||||
|
];
|
||||||
return actions;
|
}, [user]);
|
||||||
}, []);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
<>
|
||||||
|
{newParameter.modal}
|
||||||
|
{editParameter.modal}
|
||||||
|
{deleteParameter.modal}
|
||||||
<InvenTreeTable
|
<InvenTreeTable
|
||||||
url={apiUrl(ApiEndpoints.part_parameter_list)}
|
url={apiUrl(ApiEndpoints.part_parameter_list)}
|
||||||
tableState={table}
|
tableState={table}
|
||||||
@ -195,5 +196,6 @@ export function PartParameterTable({ partId }: { partId: any }) {
|
|||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -1,18 +1,18 @@
|
|||||||
import { t } from '@lingui/macro';
|
import { t } from '@lingui/macro';
|
||||||
import { useCallback, useMemo } from 'react';
|
import { useCallback, useMemo, useState } from 'react';
|
||||||
|
|
||||||
import { ApiEndpoints } from '../../../enums/ApiEndpoints';
|
import { ApiEndpoints } from '../../../enums/ApiEndpoints';
|
||||||
import { UserRoles } from '../../../enums/Roles';
|
import { UserRoles } from '../../../enums/Roles';
|
||||||
import { partParameterTemplateFields } from '../../../forms/PartForms';
|
|
||||||
import {
|
import {
|
||||||
openCreateApiForm,
|
useCreateApiFormModal,
|
||||||
openDeleteApiForm,
|
useDeleteApiFormModal,
|
||||||
openEditApiForm
|
useEditApiFormModal
|
||||||
} from '../../../functions/forms';
|
} from '../../../hooks/UseForm';
|
||||||
import { useTable } from '../../../hooks/UseTable';
|
import { useTable } from '../../../hooks/UseTable';
|
||||||
import { apiUrl } from '../../../states/ApiState';
|
import { apiUrl } from '../../../states/ApiState';
|
||||||
import { useUserState } from '../../../states/UserState';
|
import { useUserState } from '../../../states/UserState';
|
||||||
import { AddItemButton } from '../../buttons/AddItemButton';
|
import { AddItemButton } from '../../buttons/AddItemButton';
|
||||||
|
import { ApiFormFieldSet } from '../../forms/fields/ApiFormField';
|
||||||
import { TableColumn } from '../Column';
|
import { TableColumn } from '../Column';
|
||||||
import { DescriptionColumn } from '../ColumnRenderers';
|
import { DescriptionColumn } from '../ColumnRenderers';
|
||||||
import { TableFilter } from '../Filter';
|
import { TableFilter } from '../Filter';
|
||||||
@ -69,6 +69,42 @@ export default function PartParameterTemplateTable() {
|
|||||||
];
|
];
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
const partParameterTemplateFields: ApiFormFieldSet = useMemo(() => {
|
||||||
|
return {
|
||||||
|
name: {},
|
||||||
|
description: {},
|
||||||
|
units: {},
|
||||||
|
choices: {},
|
||||||
|
checkbox: {}
|
||||||
|
};
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const newTemplate = useCreateApiFormModal({
|
||||||
|
url: ApiEndpoints.part_parameter_template_list,
|
||||||
|
title: t`Add Parameter Template`,
|
||||||
|
fields: partParameterTemplateFields,
|
||||||
|
onFormSuccess: table.refreshTable
|
||||||
|
});
|
||||||
|
|
||||||
|
const [selectedTemplate, setSelectedTemplate] = useState<number | undefined>(
|
||||||
|
undefined
|
||||||
|
);
|
||||||
|
|
||||||
|
const editTemplate = useEditApiFormModal({
|
||||||
|
url: ApiEndpoints.part_parameter_template_list,
|
||||||
|
pk: selectedTemplate,
|
||||||
|
title: t`Edit Parameter Template`,
|
||||||
|
fields: partParameterTemplateFields,
|
||||||
|
onFormSuccess: table.refreshTable
|
||||||
|
});
|
||||||
|
|
||||||
|
const deleteTemplate = useDeleteApiFormModal({
|
||||||
|
url: ApiEndpoints.part_parameter_template_list,
|
||||||
|
pk: selectedTemplate,
|
||||||
|
title: t`Delete Parameter Template`,
|
||||||
|
onFormSuccess: table.refreshTable
|
||||||
|
});
|
||||||
|
|
||||||
// Callback for row actions
|
// Callback for row actions
|
||||||
const rowActions = useCallback(
|
const rowActions = useCallback(
|
||||||
(record: any) => {
|
(record: any) => {
|
||||||
@ -76,27 +112,15 @@ export default function PartParameterTemplateTable() {
|
|||||||
RowEditAction({
|
RowEditAction({
|
||||||
hidden: !user.hasChangeRole(UserRoles.part),
|
hidden: !user.hasChangeRole(UserRoles.part),
|
||||||
onClick: () => {
|
onClick: () => {
|
||||||
openEditApiForm({
|
setSelectedTemplate(record.pk);
|
||||||
url: ApiEndpoints.part_parameter_template_list,
|
editTemplate.open();
|
||||||
pk: record.pk,
|
|
||||||
title: t`Edit Parameter Template`,
|
|
||||||
fields: partParameterTemplateFields(),
|
|
||||||
successMessage: t`Parameter template updated`,
|
|
||||||
onFormSuccess: table.refreshTable
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
}),
|
}),
|
||||||
RowDeleteAction({
|
RowDeleteAction({
|
||||||
hidden: !user.hasDeleteRole(UserRoles.part),
|
hidden: !user.hasDeleteRole(UserRoles.part),
|
||||||
onClick: () => {
|
onClick: () => {
|
||||||
openDeleteApiForm({
|
setSelectedTemplate(record.pk);
|
||||||
url: ApiEndpoints.part_parameter_template_list,
|
deleteTemplate.open();
|
||||||
pk: record.pk,
|
|
||||||
title: t`Delete Parameter Template`,
|
|
||||||
successMessage: t`Parameter template deleted`,
|
|
||||||
onFormSuccess: table.refreshTable,
|
|
||||||
preFormWarning: t`Are you sure you want to remove this parameter template?`
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
];
|
];
|
||||||
@ -104,27 +128,21 @@ export default function PartParameterTemplateTable() {
|
|||||||
[user]
|
[user]
|
||||||
);
|
);
|
||||||
|
|
||||||
const addParameterTemplate = useCallback(() => {
|
|
||||||
openCreateApiForm({
|
|
||||||
url: ApiEndpoints.part_parameter_template_list,
|
|
||||||
title: t`Create Parameter Template`,
|
|
||||||
fields: partParameterTemplateFields(),
|
|
||||||
successMessage: t`Parameter template created`,
|
|
||||||
onFormSuccess: table.refreshTable
|
|
||||||
});
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const tableActions = useMemo(() => {
|
const tableActions = useMemo(() => {
|
||||||
return [
|
return [
|
||||||
<AddItemButton
|
<AddItemButton
|
||||||
tooltip={t`Add parameter template`}
|
tooltip={t`Add parameter template`}
|
||||||
onClick={addParameterTemplate}
|
onClick={() => newTemplate.open()}
|
||||||
disabled={!user.hasAddRole(UserRoles.part)}
|
disabled={!user.hasAddRole(UserRoles.part)}
|
||||||
/>
|
/>
|
||||||
];
|
];
|
||||||
}, [user]);
|
}, [user]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
<>
|
||||||
|
{newTemplate.modal}
|
||||||
|
{editTemplate.modal}
|
||||||
|
{deleteTemplate.modal}
|
||||||
<InvenTreeTable
|
<InvenTreeTable
|
||||||
url={apiUrl(ApiEndpoints.part_parameter_template_list)}
|
url={apiUrl(ApiEndpoints.part_parameter_template_list)}
|
||||||
tableState={table}
|
tableState={table}
|
||||||
@ -135,5 +153,6 @@ export default function PartParameterTemplateTable() {
|
|||||||
tableActions: tableActions
|
tableActions: tableActions
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -1,18 +1,18 @@
|
|||||||
import { t } from '@lingui/macro';
|
import { t } from '@lingui/macro';
|
||||||
import { useCallback, useMemo } from 'react';
|
import { useCallback, useMemo, useState } from 'react';
|
||||||
|
|
||||||
import { ApiEndpoints } from '../../../enums/ApiEndpoints';
|
import { ApiEndpoints } from '../../../enums/ApiEndpoints';
|
||||||
import { UserRoles } from '../../../enums/Roles';
|
import { UserRoles } from '../../../enums/Roles';
|
||||||
import { partTestTemplateFields } from '../../../forms/PartForms';
|
|
||||||
import {
|
import {
|
||||||
openCreateApiForm,
|
useCreateApiFormModal,
|
||||||
openDeleteApiForm,
|
useDeleteApiFormModal,
|
||||||
openEditApiForm
|
useEditApiFormModal
|
||||||
} from '../../../functions/forms';
|
} from '../../../hooks/UseForm';
|
||||||
import { useTable } from '../../../hooks/UseTable';
|
import { useTable } from '../../../hooks/UseTable';
|
||||||
import { apiUrl } from '../../../states/ApiState';
|
import { apiUrl } from '../../../states/ApiState';
|
||||||
import { useUserState } from '../../../states/UserState';
|
import { useUserState } from '../../../states/UserState';
|
||||||
import { AddItemButton } from '../../buttons/AddItemButton';
|
import { AddItemButton } from '../../buttons/AddItemButton';
|
||||||
|
import { ApiFormFieldSet } from '../../forms/fields/ApiFormField';
|
||||||
import { TableColumn } from '../Column';
|
import { TableColumn } from '../Column';
|
||||||
import { BooleanColumn, DescriptionColumn } from '../ColumnRenderers';
|
import { BooleanColumn, DescriptionColumn } from '../ColumnRenderers';
|
||||||
import { TableFilter } from '../Filter';
|
import { TableFilter } from '../Filter';
|
||||||
@ -69,6 +69,48 @@ export default function PartTestTemplateTable({ partId }: { partId: number }) {
|
|||||||
];
|
];
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
const partTestTemplateFields: ApiFormFieldSet = useMemo(() => {
|
||||||
|
return {
|
||||||
|
part: {
|
||||||
|
hidden: true
|
||||||
|
},
|
||||||
|
test_name: {},
|
||||||
|
description: {},
|
||||||
|
required: {},
|
||||||
|
requires_value: {},
|
||||||
|
requires_attachment: {}
|
||||||
|
};
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const newTestTemplate = useCreateApiFormModal({
|
||||||
|
url: ApiEndpoints.part_test_template_list,
|
||||||
|
title: t`Add Test Template`,
|
||||||
|
fields: partTestTemplateFields,
|
||||||
|
initialData: {
|
||||||
|
part: partId
|
||||||
|
},
|
||||||
|
onFormSuccess: table.refreshTable
|
||||||
|
});
|
||||||
|
|
||||||
|
const [selectedTest, setSelectedTest] = useState<number | undefined>(
|
||||||
|
undefined
|
||||||
|
);
|
||||||
|
|
||||||
|
const editTestTemplate = useEditApiFormModal({
|
||||||
|
url: ApiEndpoints.part_test_template_list,
|
||||||
|
pk: selectedTest,
|
||||||
|
title: t`Edit Test Template`,
|
||||||
|
fields: partTestTemplateFields,
|
||||||
|
onFormSuccess: table.refreshTable
|
||||||
|
});
|
||||||
|
|
||||||
|
const deleteTestTemplate = useDeleteApiFormModal({
|
||||||
|
url: ApiEndpoints.part_test_template_list,
|
||||||
|
pk: selectedTest,
|
||||||
|
title: t`Delete Test Template`,
|
||||||
|
onFormSuccess: table.refreshTable
|
||||||
|
});
|
||||||
|
|
||||||
const rowActions = useCallback(
|
const rowActions = useCallback(
|
||||||
(record: any) => {
|
(record: any) => {
|
||||||
let can_edit = user.hasChangeRole(UserRoles.part);
|
let can_edit = user.hasChangeRole(UserRoles.part);
|
||||||
@ -78,26 +120,15 @@ export default function PartTestTemplateTable({ partId }: { partId: number }) {
|
|||||||
RowEditAction({
|
RowEditAction({
|
||||||
hidden: !can_edit,
|
hidden: !can_edit,
|
||||||
onClick: () => {
|
onClick: () => {
|
||||||
openEditApiForm({
|
setSelectedTest(record.pk);
|
||||||
url: ApiEndpoints.part_test_template_list,
|
editTestTemplate.open();
|
||||||
pk: record.pk,
|
|
||||||
title: t`Edit Test Template`,
|
|
||||||
fields: partTestTemplateFields(),
|
|
||||||
successMessage: t`Template updated`,
|
|
||||||
onFormSuccess: table.refreshTable
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
}),
|
}),
|
||||||
RowDeleteAction({
|
RowDeleteAction({
|
||||||
hidden: !can_delete,
|
hidden: !can_delete,
|
||||||
onClick: () => {
|
onClick: () => {
|
||||||
openDeleteApiForm({
|
setSelectedTest(record.pk);
|
||||||
url: ApiEndpoints.part_test_template_list,
|
deleteTestTemplate.open();
|
||||||
pk: record.pk,
|
|
||||||
title: t`Delete Test Template`,
|
|
||||||
successMessage: t`Test Template deleted`,
|
|
||||||
onFormSuccess: table.refreshTable
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
];
|
];
|
||||||
@ -105,33 +136,23 @@ export default function PartTestTemplateTable({ partId }: { partId: number }) {
|
|||||||
[user]
|
[user]
|
||||||
);
|
);
|
||||||
|
|
||||||
const addTestTemplate = useCallback(() => {
|
|
||||||
let fields = partTestTemplateFields();
|
|
||||||
|
|
||||||
fields['part'].value = partId;
|
|
||||||
|
|
||||||
openCreateApiForm({
|
|
||||||
url: ApiEndpoints.part_test_template_list,
|
|
||||||
title: t`Create Test Template`,
|
|
||||||
fields: fields,
|
|
||||||
successMessage: t`Template created`,
|
|
||||||
onFormSuccess: table.refreshTable
|
|
||||||
});
|
|
||||||
}, [partId]);
|
|
||||||
|
|
||||||
const tableActions = useMemo(() => {
|
const tableActions = useMemo(() => {
|
||||||
let can_add = user.hasAddRole(UserRoles.part);
|
let can_add = user.hasAddRole(UserRoles.part);
|
||||||
|
|
||||||
return [
|
return [
|
||||||
<AddItemButton
|
<AddItemButton
|
||||||
tooltip={t`Add Test Template`}
|
tooltip={t`Add Test Template`}
|
||||||
onClick={addTestTemplate}
|
onClick={() => newTestTemplate.open()}
|
||||||
disabled={!can_add}
|
disabled={!can_add}
|
||||||
/>
|
/>
|
||||||
];
|
];
|
||||||
}, [user]);
|
}, [user]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
<>
|
||||||
|
{newTestTemplate.modal}
|
||||||
|
{editTestTemplate.modal}
|
||||||
|
{deleteTestTemplate.modal}
|
||||||
<InvenTreeTable
|
<InvenTreeTable
|
||||||
url={apiUrl(ApiEndpoints.part_test_template_list)}
|
url={apiUrl(ApiEndpoints.part_test_template_list)}
|
||||||
tableState={table}
|
tableState={table}
|
||||||
@ -145,5 +166,6 @@ export default function PartTestTemplateTable({ partId }: { partId: number }) {
|
|||||||
rowActions: rowActions
|
rowActions: rowActions
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -1,15 +1,19 @@
|
|||||||
import { t } from '@lingui/macro';
|
import { t } from '@lingui/macro';
|
||||||
import { ActionIcon, Group, Text, Tooltip } from '@mantine/core';
|
import { Group, Text } from '@mantine/core';
|
||||||
import { IconLayersLinked } from '@tabler/icons-react';
|
import { ReactNode, useCallback, useMemo, useState } from 'react';
|
||||||
import { ReactNode, useCallback, useMemo } from 'react';
|
|
||||||
import { useNavigate } from 'react-router-dom';
|
import { useNavigate } from 'react-router-dom';
|
||||||
|
|
||||||
import { ApiEndpoints } from '../../../enums/ApiEndpoints';
|
import { ApiEndpoints } from '../../../enums/ApiEndpoints';
|
||||||
import { UserRoles } from '../../../enums/Roles';
|
import { UserRoles } from '../../../enums/Roles';
|
||||||
import { openCreateApiForm, openDeleteApiForm } from '../../../functions/forms';
|
import {
|
||||||
|
useCreateApiFormModal,
|
||||||
|
useDeleteApiFormModal
|
||||||
|
} from '../../../hooks/UseForm';
|
||||||
import { useTable } from '../../../hooks/UseTable';
|
import { useTable } from '../../../hooks/UseTable';
|
||||||
import { apiUrl } from '../../../states/ApiState';
|
import { apiUrl } from '../../../states/ApiState';
|
||||||
import { useUserState } from '../../../states/UserState';
|
import { useUserState } from '../../../states/UserState';
|
||||||
|
import { AddItemButton } from '../../buttons/AddItemButton';
|
||||||
|
import { ApiFormFieldSet } from '../../forms/fields/ApiFormField';
|
||||||
import { Thumbnail } from '../../images/Thumbnail';
|
import { Thumbnail } from '../../images/Thumbnail';
|
||||||
import { TableColumn } from '../Column';
|
import { TableColumn } from '../Column';
|
||||||
import { InvenTreeTable } from '../InvenTreeTable';
|
import { InvenTreeTable } from '../InvenTreeTable';
|
||||||
@ -66,55 +70,54 @@ export function RelatedPartTable({ partId }: { partId: number }): ReactNode {
|
|||||||
];
|
];
|
||||||
}, [partId]);
|
}, [partId]);
|
||||||
|
|
||||||
const addRelatedPart = useCallback(() => {
|
const relatedPartFields: ApiFormFieldSet = useMemo(() => {
|
||||||
openCreateApiForm({
|
return {
|
||||||
title: t`Add Related Part`,
|
|
||||||
url: ApiEndpoints.related_part_list,
|
|
||||||
fields: {
|
|
||||||
part_1: {
|
part_1: {
|
||||||
hidden: true,
|
hidden: true
|
||||||
value: partId
|
|
||||||
},
|
},
|
||||||
part_2: {
|
part_2: {}
|
||||||
label: t`Related Part`
|
};
|
||||||
}
|
|
||||||
},
|
|
||||||
successMessage: t`Related part added`,
|
|
||||||
onFormSuccess: table.refreshTable
|
|
||||||
});
|
|
||||||
}, [partId]);
|
|
||||||
|
|
||||||
const customActions: ReactNode[] = useMemo(() => {
|
|
||||||
// TODO: Hide if user does not have permission to edit parts
|
|
||||||
let actions = [];
|
|
||||||
|
|
||||||
actions.push(
|
|
||||||
<Tooltip label={t`Add related part`}>
|
|
||||||
<ActionIcon radius="sm" onClick={addRelatedPart}>
|
|
||||||
<IconLayersLinked />
|
|
||||||
</ActionIcon>
|
|
||||||
</Tooltip>
|
|
||||||
);
|
|
||||||
|
|
||||||
return actions;
|
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
// Generate row actions
|
const newRelatedPart = useCreateApiFormModal({
|
||||||
// TODO: Hide if user does not have permission to edit parts
|
url: ApiEndpoints.related_part_list,
|
||||||
|
title: t`Add Related Part`,
|
||||||
|
fields: relatedPartFields,
|
||||||
|
initialData: {
|
||||||
|
part_1: partId
|
||||||
|
},
|
||||||
|
onFormSuccess: table.refreshTable
|
||||||
|
});
|
||||||
|
|
||||||
|
const [selectedRelatedPart, setSelectedRelatedPart] = useState<
|
||||||
|
number | undefined
|
||||||
|
>(undefined);
|
||||||
|
|
||||||
|
const deleteRelatedPart = useDeleteApiFormModal({
|
||||||
|
url: ApiEndpoints.related_part_list,
|
||||||
|
pk: selectedRelatedPart,
|
||||||
|
title: t`Delete Related Part`,
|
||||||
|
onFormSuccess: table.refreshTable
|
||||||
|
});
|
||||||
|
|
||||||
|
const tableActions: ReactNode[] = useMemo(() => {
|
||||||
|
return [
|
||||||
|
<AddItemButton
|
||||||
|
tooltip={t`Add related part`}
|
||||||
|
hidden={!user.hasAddRole(UserRoles.part)}
|
||||||
|
onClick={() => newRelatedPart.open()}
|
||||||
|
/>
|
||||||
|
];
|
||||||
|
}, [user]);
|
||||||
|
|
||||||
const rowActions = useCallback(
|
const rowActions = useCallback(
|
||||||
(record: any) => {
|
(record: any) => {
|
||||||
return [
|
return [
|
||||||
RowDeleteAction({
|
RowDeleteAction({
|
||||||
hidden: !user.hasDeleteRole(UserRoles.part),
|
hidden: !user.hasDeleteRole(UserRoles.part),
|
||||||
onClick: () => {
|
onClick: () => {
|
||||||
openDeleteApiForm({
|
setSelectedRelatedPart(record.pk);
|
||||||
url: ApiEndpoints.related_part_list,
|
deleteRelatedPart.open();
|
||||||
pk: record.pk,
|
|
||||||
title: t`Delete Related Part`,
|
|
||||||
successMessage: t`Related part deleted`,
|
|
||||||
preFormWarning: t`Are you sure you want to remove this relationship?`,
|
|
||||||
onFormSuccess: table.refreshTable
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
];
|
];
|
||||||
@ -123,6 +126,9 @@ export function RelatedPartTable({ partId }: { partId: number }): ReactNode {
|
|||||||
);
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
<>
|
||||||
|
{newRelatedPart.modal}
|
||||||
|
{deleteRelatedPart.modal}
|
||||||
<InvenTreeTable
|
<InvenTreeTable
|
||||||
url={apiUrl(ApiEndpoints.related_part_list)}
|
url={apiUrl(ApiEndpoints.related_part_list)}
|
||||||
tableState={table}
|
tableState={table}
|
||||||
@ -133,8 +139,9 @@ export function RelatedPartTable({ partId }: { partId: number }): ReactNode {
|
|||||||
category_detail: true
|
category_detail: true
|
||||||
},
|
},
|
||||||
rowActions: rowActions,
|
rowActions: rowActions,
|
||||||
tableActions: customActions
|
tableActions: tableActions
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
import { t } from '@lingui/macro';
|
import { t } from '@lingui/macro';
|
||||||
import { Text } from '@mantine/core';
|
import { Text } from '@mantine/core';
|
||||||
import { IconSquareArrowRight } from '@tabler/icons-react';
|
import { IconSquareArrowRight } from '@tabler/icons-react';
|
||||||
import { useCallback, useMemo } from 'react';
|
import { useCallback, useMemo, useState } from 'react';
|
||||||
import { useNavigate } from 'react-router-dom';
|
import { useNavigate } from 'react-router-dom';
|
||||||
|
|
||||||
import { ProgressBar } from '../../../components/items/ProgressBar';
|
import { ProgressBar } from '../../../components/items/ProgressBar';
|
||||||
@ -9,8 +9,12 @@ import { ApiEndpoints } from '../../../enums/ApiEndpoints';
|
|||||||
import { ModelType } from '../../../enums/ModelType';
|
import { ModelType } from '../../../enums/ModelType';
|
||||||
import { UserRoles } from '../../../enums/Roles';
|
import { UserRoles } from '../../../enums/Roles';
|
||||||
import { purchaseOrderLineItemFields } from '../../../forms/PurchaseOrderForms';
|
import { purchaseOrderLineItemFields } from '../../../forms/PurchaseOrderForms';
|
||||||
import { openCreateApiForm, openEditApiForm } from '../../../functions/forms';
|
|
||||||
import { getDetailUrl } from '../../../functions/urls';
|
import { getDetailUrl } from '../../../functions/urls';
|
||||||
|
import {
|
||||||
|
useCreateApiFormModal,
|
||||||
|
useDeleteApiFormModal,
|
||||||
|
useEditApiFormModal
|
||||||
|
} from '../../../hooks/UseForm';
|
||||||
import { useTable } from '../../../hooks/UseTable';
|
import { useTable } from '../../../hooks/UseTable';
|
||||||
import { apiUrl } from '../../../states/ApiState';
|
import { apiUrl } from '../../../states/ApiState';
|
||||||
import { useUserState } from '../../../states/UserState';
|
import { useUserState } from '../../../states/UserState';
|
||||||
@ -47,52 +51,6 @@ export function PurchaseOrderLineItemTable({
|
|||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
const user = useUserState();
|
const user = useUserState();
|
||||||
|
|
||||||
const rowActions = useCallback(
|
|
||||||
(record: any) => {
|
|
||||||
let received = (record?.received ?? 0) >= (record?.quantity ?? 0);
|
|
||||||
|
|
||||||
return [
|
|
||||||
{
|
|
||||||
hidden: received,
|
|
||||||
title: t`Receive line item`,
|
|
||||||
icon: <IconSquareArrowRight />,
|
|
||||||
color: 'green'
|
|
||||||
},
|
|
||||||
RowEditAction({
|
|
||||||
hidden: !user.hasAddRole(UserRoles.purchase_order),
|
|
||||||
onClick: () => {
|
|
||||||
let supplier = record?.supplier_part_detail?.supplier;
|
|
||||||
|
|
||||||
if (!supplier) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
let fields = purchaseOrderLineItemFields({
|
|
||||||
supplierId: supplier,
|
|
||||||
create: false
|
|
||||||
});
|
|
||||||
|
|
||||||
openEditApiForm({
|
|
||||||
url: ApiEndpoints.purchase_order_line_list,
|
|
||||||
pk: record.pk,
|
|
||||||
title: t`Edit Line Item`,
|
|
||||||
fields: fields,
|
|
||||||
onFormSuccess: table.refreshTable,
|
|
||||||
successMessage: t`Line item updated`
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}),
|
|
||||||
RowDuplicateAction({
|
|
||||||
hidden: !user.hasAddRole(UserRoles.purchase_order)
|
|
||||||
}),
|
|
||||||
RowDeleteAction({
|
|
||||||
hidden: !user.hasDeleteRole(UserRoles.purchase_order)
|
|
||||||
})
|
|
||||||
];
|
|
||||||
},
|
|
||||||
[orderId, user]
|
|
||||||
);
|
|
||||||
|
|
||||||
const tableColumns = useMemo(() => {
|
const tableColumns = useMemo(() => {
|
||||||
return [
|
return [
|
||||||
{
|
{
|
||||||
@ -220,18 +178,67 @@ export function PurchaseOrderLineItemTable({
|
|||||||
];
|
];
|
||||||
}, [orderId, user]);
|
}, [orderId, user]);
|
||||||
|
|
||||||
const addLine = useCallback(() => {
|
const newLine = useCreateApiFormModal({
|
||||||
openCreateApiForm({
|
|
||||||
url: ApiEndpoints.purchase_order_line_list,
|
url: ApiEndpoints.purchase_order_line_list,
|
||||||
title: t`Add Line Item`,
|
title: t`Add Line Item`,
|
||||||
fields: purchaseOrderLineItemFields({
|
fields: purchaseOrderLineItemFields(),
|
||||||
create: true,
|
initialData: {
|
||||||
orderId: orderId
|
order: orderId
|
||||||
}),
|
},
|
||||||
onFormSuccess: table.refreshTable,
|
onFormSuccess: table.refreshTable
|
||||||
successMessage: t`Line item added`
|
|
||||||
});
|
});
|
||||||
}, [orderId]);
|
|
||||||
|
const [selectedLine, setSelectedLine] = useState<number | undefined>(
|
||||||
|
undefined
|
||||||
|
);
|
||||||
|
|
||||||
|
const editLine = useEditApiFormModal({
|
||||||
|
url: ApiEndpoints.purchase_order_line_list,
|
||||||
|
pk: selectedLine,
|
||||||
|
title: t`Edit Line Item`,
|
||||||
|
fields: purchaseOrderLineItemFields(),
|
||||||
|
onFormSuccess: table.refreshTable
|
||||||
|
});
|
||||||
|
|
||||||
|
const deleteLine = useDeleteApiFormModal({
|
||||||
|
url: ApiEndpoints.purchase_order_line_list,
|
||||||
|
pk: selectedLine,
|
||||||
|
title: t`Delete Line Item`,
|
||||||
|
onFormSuccess: table.refreshTable
|
||||||
|
});
|
||||||
|
|
||||||
|
const rowActions = useCallback(
|
||||||
|
(record: any) => {
|
||||||
|
let received = (record?.received ?? 0) >= (record?.quantity ?? 0);
|
||||||
|
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
hidden: received,
|
||||||
|
title: t`Receive line item`,
|
||||||
|
icon: <IconSquareArrowRight />,
|
||||||
|
color: 'green'
|
||||||
|
},
|
||||||
|
RowEditAction({
|
||||||
|
hidden: !user.hasChangeRole(UserRoles.purchase_order),
|
||||||
|
onClick: () => {
|
||||||
|
setSelectedLine(record.pk);
|
||||||
|
editLine.open();
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
RowDuplicateAction({
|
||||||
|
hidden: !user.hasAddRole(UserRoles.purchase_order)
|
||||||
|
}),
|
||||||
|
RowDeleteAction({
|
||||||
|
hidden: !user.hasDeleteRole(UserRoles.purchase_order),
|
||||||
|
onClick: () => {
|
||||||
|
setSelectedLine(record.pk);
|
||||||
|
deleteLine.open();
|
||||||
|
}
|
||||||
|
})
|
||||||
|
];
|
||||||
|
},
|
||||||
|
[orderId, user]
|
||||||
|
);
|
||||||
|
|
||||||
// Custom table actions
|
// Custom table actions
|
||||||
const tableActions = useMemo(() => {
|
const tableActions = useMemo(() => {
|
||||||
@ -239,7 +246,7 @@ export function PurchaseOrderLineItemTable({
|
|||||||
<AddItemButton
|
<AddItemButton
|
||||||
key="add-line-item"
|
key="add-line-item"
|
||||||
tooltip={t`Add line item`}
|
tooltip={t`Add line item`}
|
||||||
onClick={addLine}
|
onClick={() => newLine.open()}
|
||||||
hidden={!user?.hasAddRole(UserRoles.purchase_order)}
|
hidden={!user?.hasAddRole(UserRoles.purchase_order)}
|
||||||
/>,
|
/>,
|
||||||
<ActionButton
|
<ActionButton
|
||||||
@ -251,6 +258,10 @@ export function PurchaseOrderLineItemTable({
|
|||||||
}, [orderId, user]);
|
}, [orderId, user]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
<>
|
||||||
|
{newLine.modal}
|
||||||
|
{editLine.modal}
|
||||||
|
{deleteLine.modal}
|
||||||
<InvenTreeTable
|
<InvenTreeTable
|
||||||
url={apiUrl(ApiEndpoints.purchase_order_line_list)}
|
url={apiUrl(ApiEndpoints.purchase_order_line_list)}
|
||||||
tableState={table}
|
tableState={table}
|
||||||
@ -272,5 +283,6 @@ export function PurchaseOrderLineItemTable({
|
|||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -1,12 +1,13 @@
|
|||||||
import { t } from '@lingui/macro';
|
import { t } from '@lingui/macro';
|
||||||
import { useCallback, useMemo } from 'react';
|
import { useMemo } from 'react';
|
||||||
import { useNavigate } from 'react-router-dom';
|
import { useNavigate } from 'react-router-dom';
|
||||||
|
|
||||||
import { ApiEndpoints } from '../../../enums/ApiEndpoints';
|
import { ApiEndpoints } from '../../../enums/ApiEndpoints';
|
||||||
import { ModelType } from '../../../enums/ModelType';
|
import { ModelType } from '../../../enums/ModelType';
|
||||||
import { UserRoles } from '../../../enums/Roles';
|
import { UserRoles } from '../../../enums/Roles';
|
||||||
import { notYetImplemented } from '../../../functions/notifications';
|
import { purchaseOrderFields } from '../../../forms/PurchaseOrderForms';
|
||||||
import { getDetailUrl } from '../../../functions/urls';
|
import { getDetailUrl } from '../../../functions/urls';
|
||||||
|
import { useCreateApiFormModal } from '../../../hooks/UseForm';
|
||||||
import { useTable } from '../../../hooks/UseTable';
|
import { useTable } from '../../../hooks/UseTable';
|
||||||
import { apiUrl } from '../../../states/ApiState';
|
import { apiUrl } from '../../../states/ApiState';
|
||||||
import { useUserState } from '../../../states/UserState';
|
import { useUserState } from '../../../states/UserState';
|
||||||
@ -34,7 +35,13 @@ import { InvenTreeTable } from '../InvenTreeTable';
|
|||||||
/**
|
/**
|
||||||
* Display a table of purchase orders
|
* Display a table of purchase orders
|
||||||
*/
|
*/
|
||||||
export function PurchaseOrderTable({ params }: { params?: any }) {
|
export function PurchaseOrderTable({
|
||||||
|
supplierId,
|
||||||
|
supplierPartId
|
||||||
|
}: {
|
||||||
|
supplierId?: number;
|
||||||
|
supplierPartId?: number;
|
||||||
|
}) {
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
|
|
||||||
const table = useTable('purchase-order');
|
const table = useTable('purchase-order');
|
||||||
@ -56,10 +63,6 @@ export function PurchaseOrderTable({ params }: { params?: any }) {
|
|||||||
];
|
];
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
// TODO: Row actions
|
|
||||||
|
|
||||||
// TODO: Table actions (e.g. create new purchase order)
|
|
||||||
|
|
||||||
const tableColumns = useMemo(() => {
|
const tableColumns = useMemo(() => {
|
||||||
return [
|
return [
|
||||||
{
|
{
|
||||||
@ -100,29 +103,44 @@ export function PurchaseOrderTable({ params }: { params?: any }) {
|
|||||||
];
|
];
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const addPurchaseOrder = useCallback(() => {
|
const newPurchaseOrder = useCreateApiFormModal({
|
||||||
notYetImplemented();
|
url: ApiEndpoints.purchase_order_list,
|
||||||
}, []);
|
title: t`Add Purchase Order`,
|
||||||
|
fields: purchaseOrderFields(),
|
||||||
|
initialData: {
|
||||||
|
supplier: supplierId
|
||||||
|
},
|
||||||
|
onFormSuccess: (response) => {
|
||||||
|
if (response.pk) {
|
||||||
|
navigate(getDetailUrl(ModelType.purchaseorder, response.pk));
|
||||||
|
} else {
|
||||||
|
table.refreshTable();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
const tableActions = useMemo(() => {
|
const tableActions = useMemo(() => {
|
||||||
return [
|
return [
|
||||||
<AddItemButton
|
<AddItemButton
|
||||||
tooltip={t`Add Purchase Order`}
|
tooltip={t`Add Purchase Order`}
|
||||||
onClick={addPurchaseOrder}
|
onClick={() => newPurchaseOrder.open()}
|
||||||
hidden={!user.hasAddRole(UserRoles.purchase_order)}
|
hidden={!user.hasAddRole(UserRoles.purchase_order)}
|
||||||
/>
|
/>
|
||||||
];
|
];
|
||||||
}, [user]);
|
}, [user]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
<>
|
||||||
|
{newPurchaseOrder.modal}
|
||||||
<InvenTreeTable
|
<InvenTreeTable
|
||||||
url={apiUrl(ApiEndpoints.purchase_order_list)}
|
url={apiUrl(ApiEndpoints.purchase_order_list)}
|
||||||
tableState={table}
|
tableState={table}
|
||||||
columns={tableColumns}
|
columns={tableColumns}
|
||||||
props={{
|
props={{
|
||||||
params: {
|
params: {
|
||||||
...params,
|
supplier_detail: true,
|
||||||
supplier_detail: true
|
supplier: supplierId,
|
||||||
|
supplier_part: supplierPartId
|
||||||
},
|
},
|
||||||
tableFilters: tableFilters,
|
tableFilters: tableFilters,
|
||||||
tableActions: tableActions,
|
tableActions: tableActions,
|
||||||
@ -133,5 +151,6 @@ export function PurchaseOrderTable({ params }: { params?: any }) {
|
|||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -1,12 +1,13 @@
|
|||||||
import { t } from '@lingui/macro';
|
import { t } from '@lingui/macro';
|
||||||
import { useCallback, useMemo } from 'react';
|
import { useMemo } from 'react';
|
||||||
import { useNavigate } from 'react-router-dom';
|
import { useNavigate } from 'react-router-dom';
|
||||||
|
|
||||||
import { ApiEndpoints } from '../../../enums/ApiEndpoints';
|
import { ApiEndpoints } from '../../../enums/ApiEndpoints';
|
||||||
import { ModelType } from '../../../enums/ModelType';
|
import { ModelType } from '../../../enums/ModelType';
|
||||||
import { UserRoles } from '../../../enums/Roles';
|
import { UserRoles } from '../../../enums/Roles';
|
||||||
import { notYetImplemented } from '../../../functions/notifications';
|
import { salesOrderFields } from '../../../forms/SalesOrderForms';
|
||||||
import { getDetailUrl } from '../../../functions/urls';
|
import { getDetailUrl } from '../../../functions/urls';
|
||||||
|
import { useCreateApiFormModal } from '../../../hooks/UseForm';
|
||||||
import { useTable } from '../../../hooks/UseTable';
|
import { useTable } from '../../../hooks/UseTable';
|
||||||
import { apiUrl } from '../../../states/ApiState';
|
import { apiUrl } from '../../../states/ApiState';
|
||||||
import { useUserState } from '../../../states/UserState';
|
import { useUserState } from '../../../states/UserState';
|
||||||
@ -31,7 +32,13 @@ import {
|
|||||||
} from '../Filter';
|
} from '../Filter';
|
||||||
import { InvenTreeTable } from '../InvenTreeTable';
|
import { InvenTreeTable } from '../InvenTreeTable';
|
||||||
|
|
||||||
export function SalesOrderTable({ params }: { params?: any }) {
|
export function SalesOrderTable({
|
||||||
|
partId,
|
||||||
|
customerId
|
||||||
|
}: {
|
||||||
|
partId?: number;
|
||||||
|
customerId?: number;
|
||||||
|
}) {
|
||||||
const table = useTable('sales-order');
|
const table = useTable('sales-order');
|
||||||
const user = useUserState();
|
const user = useUserState();
|
||||||
|
|
||||||
@ -53,9 +60,31 @@ export function SalesOrderTable({ params }: { params?: any }) {
|
|||||||
];
|
];
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
// TODO: Row actions
|
const newSalesOrder = useCreateApiFormModal({
|
||||||
|
url: ApiEndpoints.sales_order_list,
|
||||||
|
title: t`Add Sales Order`,
|
||||||
|
fields: salesOrderFields(),
|
||||||
|
initialData: {
|
||||||
|
customer: customerId
|
||||||
|
},
|
||||||
|
onFormSuccess: (response) => {
|
||||||
|
if (response.pk) {
|
||||||
|
navigate(getDetailUrl(ModelType.salesorder, response.pk));
|
||||||
|
} else {
|
||||||
|
table.refreshTable();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
// TODO: Table actions (e.g. create new sales order)
|
const tableActions = useMemo(() => {
|
||||||
|
return [
|
||||||
|
<AddItemButton
|
||||||
|
tooltip={t`Add Sales Order`}
|
||||||
|
onClick={() => newSalesOrder.open()}
|
||||||
|
hidden={!user.hasAddRole(UserRoles.sales_order)}
|
||||||
|
/>
|
||||||
|
];
|
||||||
|
}, [user]);
|
||||||
|
|
||||||
const tableColumns = useMemo(() => {
|
const tableColumns = useMemo(() => {
|
||||||
return [
|
return [
|
||||||
@ -97,29 +126,18 @@ export function SalesOrderTable({ params }: { params?: any }) {
|
|||||||
];
|
];
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const addSalesOrder = useCallback(() => {
|
|
||||||
notYetImplemented();
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const tableActions = useMemo(() => {
|
|
||||||
return [
|
|
||||||
<AddItemButton
|
|
||||||
tooltip={t`Add Sales Order`}
|
|
||||||
onClick={addSalesOrder}
|
|
||||||
hidden={!user.hasAddRole(UserRoles.sales_order)}
|
|
||||||
/>
|
|
||||||
];
|
|
||||||
}, [user]);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
<>
|
||||||
|
{newSalesOrder.modal}
|
||||||
<InvenTreeTable
|
<InvenTreeTable
|
||||||
url={apiUrl(ApiEndpoints.sales_order_list)}
|
url={apiUrl(ApiEndpoints.sales_order_list)}
|
||||||
tableState={table}
|
tableState={table}
|
||||||
columns={tableColumns}
|
columns={tableColumns}
|
||||||
props={{
|
props={{
|
||||||
params: {
|
params: {
|
||||||
...params,
|
customer_detail: true,
|
||||||
customer_detail: true
|
part: partId,
|
||||||
|
customer: customerId
|
||||||
},
|
},
|
||||||
tableFilters: tableFilters,
|
tableFilters: tableFilters,
|
||||||
tableActions: tableActions,
|
tableActions: tableActions,
|
||||||
@ -130,5 +148,6 @@ export function SalesOrderTable({ params }: { params?: any }) {
|
|||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -1,13 +1,14 @@
|
|||||||
import { t } from '@lingui/macro';
|
import { t } from '@lingui/macro';
|
||||||
import { useCallback, useMemo } from 'react';
|
import { useCallback, useMemo, useState } from 'react';
|
||||||
|
|
||||||
import { ApiEndpoints } from '../../../enums/ApiEndpoints';
|
import { ApiEndpoints } from '../../../enums/ApiEndpoints';
|
||||||
import { UserRoles } from '../../../enums/Roles';
|
import { UserRoles } from '../../../enums/Roles';
|
||||||
|
import { customUnitsFields } from '../../../forms/CommonForms';
|
||||||
import {
|
import {
|
||||||
openCreateApiForm,
|
useCreateApiFormModal,
|
||||||
openDeleteApiForm,
|
useDeleteApiFormModal,
|
||||||
openEditApiForm
|
useEditApiFormModal
|
||||||
} from '../../../functions/forms';
|
} from '../../../hooks/UseForm';
|
||||||
import { useTable } from '../../../hooks/UseTable';
|
import { useTable } from '../../../hooks/UseTable';
|
||||||
import { apiUrl } from '../../../states/ApiState';
|
import { apiUrl } from '../../../states/ApiState';
|
||||||
import { useUserState } from '../../../states/UserState';
|
import { useUserState } from '../../../states/UserState';
|
||||||
@ -47,37 +48,47 @@ export default function CustomUnitsTable() {
|
|||||||
];
|
];
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
const newUnit = useCreateApiFormModal({
|
||||||
|
url: ApiEndpoints.custom_unit_list,
|
||||||
|
title: t`Add Custom Unit`,
|
||||||
|
fields: customUnitsFields(),
|
||||||
|
onFormSuccess: table.refreshTable
|
||||||
|
});
|
||||||
|
|
||||||
|
const [selectedUnit, setSelectedUnit] = useState<number | undefined>(
|
||||||
|
undefined
|
||||||
|
);
|
||||||
|
|
||||||
|
const editUnit = useEditApiFormModal({
|
||||||
|
url: ApiEndpoints.custom_unit_list,
|
||||||
|
pk: selectedUnit,
|
||||||
|
title: t`Edit Custom Unit`,
|
||||||
|
fields: customUnitsFields(),
|
||||||
|
onFormSuccess: table.refreshTable
|
||||||
|
});
|
||||||
|
|
||||||
|
const deleteUnit = useDeleteApiFormModal({
|
||||||
|
url: ApiEndpoints.custom_unit_list,
|
||||||
|
pk: selectedUnit,
|
||||||
|
title: t`Delete Custom Unit`,
|
||||||
|
onFormSuccess: table.refreshTable
|
||||||
|
});
|
||||||
|
|
||||||
const rowActions = useCallback(
|
const rowActions = useCallback(
|
||||||
(record: any): RowAction[] => {
|
(record: any): RowAction[] => {
|
||||||
return [
|
return [
|
||||||
RowEditAction({
|
RowEditAction({
|
||||||
hidden: !user.hasChangeRole(UserRoles.admin),
|
hidden: !user.hasChangeRole(UserRoles.admin),
|
||||||
onClick: () => {
|
onClick: () => {
|
||||||
openEditApiForm({
|
setSelectedUnit(record.pk);
|
||||||
url: ApiEndpoints.custom_unit_list,
|
editUnit.open();
|
||||||
pk: record.pk,
|
|
||||||
title: t`Edit custom unit`,
|
|
||||||
fields: {
|
|
||||||
name: {},
|
|
||||||
definition: {},
|
|
||||||
symbol: {}
|
|
||||||
},
|
|
||||||
onFormSuccess: table.refreshTable,
|
|
||||||
successMessage: t`Custom unit updated`
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
}),
|
}),
|
||||||
RowDeleteAction({
|
RowDeleteAction({
|
||||||
hidden: !user.hasDeleteRole(UserRoles.admin),
|
hidden: !user.hasDeleteRole(UserRoles.admin),
|
||||||
onClick: () => {
|
onClick: () => {
|
||||||
openDeleteApiForm({
|
setSelectedUnit(record.pk);
|
||||||
url: ApiEndpoints.custom_unit_list,
|
deleteUnit.open();
|
||||||
pk: record.pk,
|
|
||||||
title: t`Delete custom unit`,
|
|
||||||
successMessage: t`Custom unit deleted`,
|
|
||||||
onFormSuccess: table.refreshTable,
|
|
||||||
preFormWarning: t`Are you sure you want to remove this custom unit?`
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
];
|
];
|
||||||
@ -85,32 +96,25 @@ export default function CustomUnitsTable() {
|
|||||||
[user]
|
[user]
|
||||||
);
|
);
|
||||||
|
|
||||||
const addCustomUnit = useCallback(() => {
|
|
||||||
openCreateApiForm({
|
|
||||||
url: ApiEndpoints.custom_unit_list,
|
|
||||||
title: t`Add custom unit`,
|
|
||||||
fields: {
|
|
||||||
name: {},
|
|
||||||
definition: {},
|
|
||||||
symbol: {}
|
|
||||||
},
|
|
||||||
successMessage: t`Custom unit created`,
|
|
||||||
onFormSuccess: table.refreshTable
|
|
||||||
});
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const tableActions = useMemo(() => {
|
const tableActions = useMemo(() => {
|
||||||
let actions = [];
|
let actions = [];
|
||||||
|
|
||||||
actions.push(
|
actions.push(
|
||||||
// TODO: Adjust actions based on user permissions
|
// TODO: Adjust actions based on user permissions
|
||||||
<AddItemButton tooltip={t`Add custom unit`} onClick={addCustomUnit} />
|
<AddItemButton
|
||||||
|
tooltip={t`Add custom unit`}
|
||||||
|
onClick={() => newUnit.open()}
|
||||||
|
/>
|
||||||
);
|
);
|
||||||
|
|
||||||
return actions;
|
return actions;
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
<>
|
||||||
|
{newUnit.modal}
|
||||||
|
{editUnit.modal}
|
||||||
|
{deleteUnit.modal}
|
||||||
<InvenTreeTable
|
<InvenTreeTable
|
||||||
url={apiUrl(ApiEndpoints.custom_unit_list)}
|
url={apiUrl(ApiEndpoints.custom_unit_list)}
|
||||||
tableState={table}
|
tableState={table}
|
||||||
@ -120,5 +124,6 @@ export default function CustomUnitsTable() {
|
|||||||
tableActions: tableActions
|
tableActions: tableActions
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -1,10 +1,13 @@
|
|||||||
import { Trans, t } from '@lingui/macro';
|
import { Trans, t } from '@lingui/macro';
|
||||||
import { Group, LoadingOverlay, Stack, Text, Title } from '@mantine/core';
|
import { Group, LoadingOverlay, Stack, Text, Title } from '@mantine/core';
|
||||||
import { useCallback, useMemo } from 'react';
|
import { useCallback, useMemo, useState } from 'react';
|
||||||
import { useNavigate } from 'react-router-dom';
|
import { useNavigate } from 'react-router-dom';
|
||||||
|
|
||||||
import { ApiEndpoints } from '../../../enums/ApiEndpoints';
|
import { ApiEndpoints } from '../../../enums/ApiEndpoints';
|
||||||
import { openCreateApiForm, openDeleteApiForm } from '../../../functions/forms';
|
import {
|
||||||
|
useCreateApiFormModal,
|
||||||
|
useDeleteApiFormModal
|
||||||
|
} from '../../../hooks/UseForm';
|
||||||
import { useInstance } from '../../../hooks/UseInstance';
|
import { useInstance } from '../../../hooks/UseInstance';
|
||||||
import { useTable } from '../../../hooks/UseTable';
|
import { useTable } from '../../../hooks/UseTable';
|
||||||
import { apiUrl } from '../../../states/ApiState';
|
import { apiUrl } from '../../../states/ApiState';
|
||||||
@ -109,28 +112,31 @@ export function GroupTable() {
|
|||||||
}),
|
}),
|
||||||
RowDeleteAction({
|
RowDeleteAction({
|
||||||
onClick: () => {
|
onClick: () => {
|
||||||
openDeleteApiForm({
|
setSelectedGroup(record.pk), deleteGroup.open();
|
||||||
url: ApiEndpoints.group_list,
|
|
||||||
pk: record.pk,
|
|
||||||
title: t`Delete group`,
|
|
||||||
successMessage: t`Group deleted`,
|
|
||||||
onFormSuccess: table.refreshTable,
|
|
||||||
preFormWarning: t`Are you sure you want to delete this group?`
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
];
|
];
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const addGroup = useCallback(() => {
|
const [selectedGroup, setSelectedGroup] = useState<number | undefined>(
|
||||||
openCreateApiForm({
|
undefined
|
||||||
|
);
|
||||||
|
|
||||||
|
const deleteGroup = useDeleteApiFormModal({
|
||||||
|
url: ApiEndpoints.group_list,
|
||||||
|
pk: selectedGroup,
|
||||||
|
title: t`Delete group`,
|
||||||
|
successMessage: t`Group deleted`,
|
||||||
|
onFormSuccess: table.refreshTable,
|
||||||
|
preFormWarning: t`Are you sure you want to delete this group?`
|
||||||
|
});
|
||||||
|
|
||||||
|
const newGroup = useCreateApiFormModal({
|
||||||
url: ApiEndpoints.group_list,
|
url: ApiEndpoints.group_list,
|
||||||
title: t`Add group`,
|
title: t`Add group`,
|
||||||
fields: { name: {} },
|
fields: { name: {} },
|
||||||
onFormSuccess: table.refreshTable,
|
onFormSuccess: table.refreshTable
|
||||||
successMessage: t`Added group`
|
|
||||||
});
|
});
|
||||||
}, []);
|
|
||||||
|
|
||||||
const tableActions = useMemo(() => {
|
const tableActions = useMemo(() => {
|
||||||
let actions = [];
|
let actions = [];
|
||||||
@ -138,7 +144,7 @@ export function GroupTable() {
|
|||||||
actions.push(
|
actions.push(
|
||||||
<AddItemButton
|
<AddItemButton
|
||||||
key={'add-group'}
|
key={'add-group'}
|
||||||
onClick={addGroup}
|
onClick={() => newGroup.open()}
|
||||||
tooltip={t`Add group`}
|
tooltip={t`Add group`}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
@ -148,6 +154,8 @@ export function GroupTable() {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
|
{newGroup.modal}
|
||||||
|
{deleteGroup.modal}
|
||||||
<DetailDrawer
|
<DetailDrawer
|
||||||
title={t`Edit group`}
|
title={t`Edit group`}
|
||||||
renderContent={(id) => {
|
renderContent={(id) => {
|
||||||
|
@ -1,13 +1,14 @@
|
|||||||
import { t } from '@lingui/macro';
|
import { t } from '@lingui/macro';
|
||||||
import { useCallback, useMemo } from 'react';
|
import { useCallback, useMemo, useState } from 'react';
|
||||||
|
|
||||||
import { ApiEndpoints } from '../../../enums/ApiEndpoints';
|
import { ApiEndpoints } from '../../../enums/ApiEndpoints';
|
||||||
import { UserRoles } from '../../../enums/Roles';
|
import { UserRoles } from '../../../enums/Roles';
|
||||||
|
import { projectCodeFields } from '../../../forms/CommonForms';
|
||||||
import {
|
import {
|
||||||
openCreateApiForm,
|
useCreateApiFormModal,
|
||||||
openDeleteApiForm,
|
useDeleteApiFormModal,
|
||||||
openEditApiForm
|
useEditApiFormModal
|
||||||
} from '../../../functions/forms';
|
} from '../../../hooks/UseForm';
|
||||||
import { useTable } from '../../../hooks/UseTable';
|
import { useTable } from '../../../hooks/UseTable';
|
||||||
import { apiUrl } from '../../../states/ApiState';
|
import { apiUrl } from '../../../states/ApiState';
|
||||||
import { useUserState } from '../../../states/UserState';
|
import { useUserState } from '../../../states/UserState';
|
||||||
@ -37,37 +38,47 @@ export default function ProjectCodeTable() {
|
|||||||
];
|
];
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
const newProjectCode = useCreateApiFormModal({
|
||||||
|
url: ApiEndpoints.project_code_list,
|
||||||
|
title: t`Add Project Code`,
|
||||||
|
fields: projectCodeFields(),
|
||||||
|
onFormSuccess: table.refreshTable
|
||||||
|
});
|
||||||
|
|
||||||
|
const [selectedProjectCode, setSelectedProjectCode] = useState<
|
||||||
|
number | undefined
|
||||||
|
>(undefined);
|
||||||
|
|
||||||
|
const editProjectCode = useEditApiFormModal({
|
||||||
|
url: ApiEndpoints.project_code_list,
|
||||||
|
pk: selectedProjectCode,
|
||||||
|
title: t`Edit Project Code`,
|
||||||
|
fields: projectCodeFields(),
|
||||||
|
onFormSuccess: table.refreshTable
|
||||||
|
});
|
||||||
|
|
||||||
|
const deleteProjectCode = useDeleteApiFormModal({
|
||||||
|
url: ApiEndpoints.project_code_list,
|
||||||
|
pk: selectedProjectCode,
|
||||||
|
title: t`Delete Project Code`,
|
||||||
|
onFormSuccess: table.refreshTable
|
||||||
|
});
|
||||||
|
|
||||||
const rowActions = useCallback(
|
const rowActions = useCallback(
|
||||||
(record: any): RowAction[] => {
|
(record: any): RowAction[] => {
|
||||||
return [
|
return [
|
||||||
RowEditAction({
|
RowEditAction({
|
||||||
hidden: !user.hasChangeRole(UserRoles.admin),
|
hidden: !user.hasChangeRole(UserRoles.admin),
|
||||||
onClick: () => {
|
onClick: () => {
|
||||||
openEditApiForm({
|
setSelectedProjectCode(record.pk);
|
||||||
url: ApiEndpoints.project_code_list,
|
editProjectCode.open();
|
||||||
pk: record.pk,
|
|
||||||
title: t`Edit project code`,
|
|
||||||
fields: {
|
|
||||||
code: {},
|
|
||||||
description: {},
|
|
||||||
responsible: {}
|
|
||||||
},
|
|
||||||
onFormSuccess: table.refreshTable,
|
|
||||||
successMessage: t`Project code updated`
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
}),
|
}),
|
||||||
RowDeleteAction({
|
RowDeleteAction({
|
||||||
hidden: !user.hasDeleteRole(UserRoles.admin),
|
hidden: !user.hasDeleteRole(UserRoles.admin),
|
||||||
onClick: () => {
|
onClick: () => {
|
||||||
openDeleteApiForm({
|
setSelectedProjectCode(record.pk);
|
||||||
url: ApiEndpoints.project_code_list,
|
deleteProjectCode.open();
|
||||||
pk: record.pk,
|
|
||||||
title: t`Delete project code`,
|
|
||||||
successMessage: t`Project code deleted`,
|
|
||||||
onFormSuccess: table.refreshTable,
|
|
||||||
preFormWarning: t`Are you sure you want to remove this project code?`
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
];
|
];
|
||||||
@ -75,31 +86,24 @@ export default function ProjectCodeTable() {
|
|||||||
[user]
|
[user]
|
||||||
);
|
);
|
||||||
|
|
||||||
const addProjectCode = useCallback(() => {
|
|
||||||
openCreateApiForm({
|
|
||||||
url: ApiEndpoints.project_code_list,
|
|
||||||
title: t`Add project code`,
|
|
||||||
fields: {
|
|
||||||
code: {},
|
|
||||||
description: {},
|
|
||||||
responsible: {}
|
|
||||||
},
|
|
||||||
onFormSuccess: table.refreshTable,
|
|
||||||
successMessage: t`Added project code`
|
|
||||||
});
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const tableActions = useMemo(() => {
|
const tableActions = useMemo(() => {
|
||||||
let actions = [];
|
let actions = [];
|
||||||
|
|
||||||
actions.push(
|
actions.push(
|
||||||
<AddItemButton onClick={addProjectCode} tooltip={t`Add project code`} />
|
<AddItemButton
|
||||||
|
onClick={() => newProjectCode.open()}
|
||||||
|
tooltip={t`Add project code`}
|
||||||
|
/>
|
||||||
);
|
);
|
||||||
|
|
||||||
return actions;
|
return actions;
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
<>
|
||||||
|
{newProjectCode.modal}
|
||||||
|
{editProjectCode.modal}
|
||||||
|
{deleteProjectCode.modal}
|
||||||
<InvenTreeTable
|
<InvenTreeTable
|
||||||
url={apiUrl(ApiEndpoints.project_code_list)}
|
url={apiUrl(ApiEndpoints.project_code_list)}
|
||||||
tableState={table}
|
tableState={table}
|
||||||
@ -109,5 +113,6 @@ export default function ProjectCodeTable() {
|
|||||||
tableActions: tableActions
|
tableActions: tableActions
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -1,13 +1,16 @@
|
|||||||
import { t } from '@lingui/macro';
|
import { t } from '@lingui/macro';
|
||||||
import { useCallback, useMemo } from 'react';
|
import { useCallback, useMemo, useState } from 'react';
|
||||||
import { useNavigate } from 'react-router-dom';
|
import { useNavigate } from 'react-router-dom';
|
||||||
|
|
||||||
import { ApiEndpoints } from '../../../enums/ApiEndpoints';
|
import { ApiEndpoints } from '../../../enums/ApiEndpoints';
|
||||||
import { ModelType } from '../../../enums/ModelType';
|
import { ModelType } from '../../../enums/ModelType';
|
||||||
import { UserRoles } from '../../../enums/Roles';
|
import { UserRoles } from '../../../enums/Roles';
|
||||||
import { stockLocationFields } from '../../../forms/StockForms';
|
import { stockLocationFields } from '../../../forms/StockForms';
|
||||||
import { openCreateApiForm, openEditApiForm } from '../../../functions/forms';
|
|
||||||
import { getDetailUrl } from '../../../functions/urls';
|
import { getDetailUrl } from '../../../functions/urls';
|
||||||
|
import {
|
||||||
|
useCreateApiFormModal,
|
||||||
|
useEditApiFormModal
|
||||||
|
} from '../../../hooks/UseForm';
|
||||||
import { useTable } from '../../../hooks/UseTable';
|
import { useTable } from '../../../hooks/UseTable';
|
||||||
import { apiUrl } from '../../../states/ApiState';
|
import { apiUrl } from '../../../states/ApiState';
|
||||||
import { useUserState } from '../../../states/UserState';
|
import { useUserState } from '../../../states/UserState';
|
||||||
@ -96,26 +99,33 @@ export function StockLocationTable({ parentId }: { parentId?: any }) {
|
|||||||
];
|
];
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const addLocation = useCallback(() => {
|
const newLocation = useCreateApiFormModal({
|
||||||
let fields = stockLocationFields({});
|
url: ApiEndpoints.stock_location_list,
|
||||||
|
|
||||||
if (parentId) {
|
|
||||||
fields['parent'].value = parentId;
|
|
||||||
}
|
|
||||||
|
|
||||||
openCreateApiForm({
|
|
||||||
url: apiUrl(ApiEndpoints.stock_location_list),
|
|
||||||
title: t`Add Stock Location`,
|
title: t`Add Stock Location`,
|
||||||
fields: fields,
|
fields: stockLocationFields({}),
|
||||||
|
initialData: {
|
||||||
|
parent: parentId
|
||||||
|
},
|
||||||
onFormSuccess(data: any) {
|
onFormSuccess(data: any) {
|
||||||
if (data.pk) {
|
if (data.pk) {
|
||||||
navigate(`/stock/location/${data.pk}`);
|
navigate(getDetailUrl(ModelType.stocklocation, data.pk));
|
||||||
} else {
|
} else {
|
||||||
table.refreshTable();
|
table.refreshTable();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}, [parentId]);
|
|
||||||
|
const [selectedLocation, setSelectedLocation] = useState<number | undefined>(
|
||||||
|
undefined
|
||||||
|
);
|
||||||
|
|
||||||
|
const editLocation = useEditApiFormModal({
|
||||||
|
url: ApiEndpoints.stock_location_list,
|
||||||
|
pk: selectedLocation,
|
||||||
|
title: t`Edit Stock Location`,
|
||||||
|
fields: stockLocationFields({}),
|
||||||
|
onFormSuccess: table.refreshTable
|
||||||
|
});
|
||||||
|
|
||||||
const tableActions = useMemo(() => {
|
const tableActions = useMemo(() => {
|
||||||
let can_add = user.hasAddRole(UserRoles.stock_location);
|
let can_add = user.hasAddRole(UserRoles.stock_location);
|
||||||
@ -123,7 +133,7 @@ export function StockLocationTable({ parentId }: { parentId?: any }) {
|
|||||||
return [
|
return [
|
||||||
<AddItemButton
|
<AddItemButton
|
||||||
tooltip={t`Add Stock Location`}
|
tooltip={t`Add Stock Location`}
|
||||||
onClick={addLocation}
|
onClick={() => newLocation.open()}
|
||||||
disabled={!can_add}
|
disabled={!can_add}
|
||||||
/>
|
/>
|
||||||
];
|
];
|
||||||
@ -137,14 +147,8 @@ export function StockLocationTable({ parentId }: { parentId?: any }) {
|
|||||||
RowEditAction({
|
RowEditAction({
|
||||||
hidden: !can_edit,
|
hidden: !can_edit,
|
||||||
onClick: () => {
|
onClick: () => {
|
||||||
openEditApiForm({
|
setSelectedLocation(record.pk);
|
||||||
url: ApiEndpoints.stock_location_list,
|
editLocation.open();
|
||||||
pk: record.pk,
|
|
||||||
title: t`Edit Stock Location`,
|
|
||||||
fields: stockLocationFields({}),
|
|
||||||
successMessage: t`Stock location updated`,
|
|
||||||
onFormSuccess: table.refreshTable
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
];
|
];
|
||||||
@ -153,6 +157,9 @@ export function StockLocationTable({ parentId }: { parentId?: any }) {
|
|||||||
);
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
<>
|
||||||
|
{newLocation.modal}
|
||||||
|
{editLocation.modal}
|
||||||
<InvenTreeTable
|
<InvenTreeTable
|
||||||
url={apiUrl(ApiEndpoints.stock_location_list)}
|
url={apiUrl(ApiEndpoints.stock_location_list)}
|
||||||
tableState={table}
|
tableState={table}
|
||||||
@ -168,8 +175,8 @@ export function StockLocationTable({ parentId }: { parentId?: any }) {
|
|||||||
onRowClick: (record) => {
|
onRowClick: (record) => {
|
||||||
navigate(getDetailUrl(ModelType.stocklocation, record.pk));
|
navigate(getDetailUrl(ModelType.stocklocation, record.pk));
|
||||||
}
|
}
|
||||||
// TODO: allow for "tree view" with cascade
|
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
21
src/frontend/src/forms/CommonForms.tsx
Normal file
21
src/frontend/src/forms/CommonForms.tsx
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
import { IconUsers } from '@tabler/icons-react';
|
||||||
|
|
||||||
|
import { ApiFormFieldSet } from '../components/forms/fields/ApiFormField';
|
||||||
|
|
||||||
|
export function projectCodeFields(): ApiFormFieldSet {
|
||||||
|
return {
|
||||||
|
code: {},
|
||||||
|
description: {},
|
||||||
|
responsible: {
|
||||||
|
icon: <IconUsers />
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function customUnitsFields(): ApiFormFieldSet {
|
||||||
|
return {
|
||||||
|
name: {},
|
||||||
|
definition: {},
|
||||||
|
symbol: {}
|
||||||
|
};
|
||||||
|
}
|
@ -1,4 +1,3 @@
|
|||||||
import { t } from '@lingui/macro';
|
|
||||||
import {
|
import {
|
||||||
IconAt,
|
IconAt,
|
||||||
IconCurrencyDollar,
|
IconCurrencyDollar,
|
||||||
@ -12,8 +11,6 @@ import {
|
|||||||
import { useEffect, useMemo, useState } from 'react';
|
import { useEffect, useMemo, useState } from 'react';
|
||||||
|
|
||||||
import { ApiFormFieldSet } from '../components/forms/fields/ApiFormField';
|
import { ApiFormFieldSet } from '../components/forms/fields/ApiFormField';
|
||||||
import { ApiEndpoints } from '../enums/ApiEndpoints';
|
|
||||||
import { openEditApiForm } from '../functions/forms';
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Field set for SupplierPart instance
|
* Field set for SupplierPart instance
|
||||||
@ -131,54 +128,3 @@ export function companyFields(): ApiFormFieldSet {
|
|||||||
is_customer: {}
|
is_customer: {}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Edit a company instance
|
|
||||||
*/
|
|
||||||
export function editCompany({
|
|
||||||
pk,
|
|
||||||
callback
|
|
||||||
}: {
|
|
||||||
pk: number;
|
|
||||||
callback?: () => void;
|
|
||||||
}) {
|
|
||||||
openEditApiForm({
|
|
||||||
title: t`Edit Company`,
|
|
||||||
url: ApiEndpoints.company_list,
|
|
||||||
pk: pk,
|
|
||||||
fields: companyFields(),
|
|
||||||
successMessage: t`Company updated`,
|
|
||||||
onFormSuccess: callback
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
export function contactFields(): ApiFormFieldSet {
|
|
||||||
return {
|
|
||||||
company: {
|
|
||||||
hidden: true
|
|
||||||
},
|
|
||||||
name: {},
|
|
||||||
phone: {},
|
|
||||||
email: {},
|
|
||||||
role: {}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
export function addressFields(): ApiFormFieldSet {
|
|
||||||
return {
|
|
||||||
company: {
|
|
||||||
hidden: true
|
|
||||||
},
|
|
||||||
title: {},
|
|
||||||
primary: {},
|
|
||||||
line1: {},
|
|
||||||
line2: {},
|
|
||||||
postal_code: {},
|
|
||||||
postal_city: {},
|
|
||||||
province: {},
|
|
||||||
country: {},
|
|
||||||
shipping_notes: {},
|
|
||||||
internal_shipping_notes: {},
|
|
||||||
link: {}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
@ -2,8 +2,6 @@ import { t } from '@lingui/macro';
|
|||||||
import { IconPackages } from '@tabler/icons-react';
|
import { IconPackages } from '@tabler/icons-react';
|
||||||
|
|
||||||
import { ApiFormFieldSet } from '../components/forms/fields/ApiFormField';
|
import { ApiFormFieldSet } from '../components/forms/fields/ApiFormField';
|
||||||
import { ApiEndpoints } from '../enums/ApiEndpoints';
|
|
||||||
import { openCreateApiForm, openEditApiForm } from '../functions/forms';
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Construct a set of fields for creating / editing a Part instance
|
* Construct a set of fields for creating / editing a Part instance
|
||||||
@ -98,39 +96,6 @@ export function partFields({
|
|||||||
return fields;
|
return fields;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Launch a dialog to create a new Part instance
|
|
||||||
*/
|
|
||||||
export function createPart() {
|
|
||||||
openCreateApiForm({
|
|
||||||
title: t`Create Part`,
|
|
||||||
url: ApiEndpoints.part_list,
|
|
||||||
successMessage: t`Part created`,
|
|
||||||
fields: partFields({})
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Launch a dialog to edit an existing Part instance
|
|
||||||
* @param part The ID of the part to edit
|
|
||||||
*/
|
|
||||||
export function editPart({
|
|
||||||
part_id,
|
|
||||||
callback
|
|
||||||
}: {
|
|
||||||
part_id: number;
|
|
||||||
callback?: () => void;
|
|
||||||
}) {
|
|
||||||
openEditApiForm({
|
|
||||||
title: t`Edit Part`,
|
|
||||||
url: ApiEndpoints.part_list,
|
|
||||||
pk: part_id,
|
|
||||||
fields: partFields({ editing: true }),
|
|
||||||
successMessage: t`Part updated`,
|
|
||||||
onFormSuccess: callback
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Construct a set of fields for creating / editing a PartCategory instance
|
* Construct a set of fields for creating / editing a PartCategory instance
|
||||||
*/
|
*/
|
||||||
@ -154,26 +119,3 @@ export function partCategoryFields({}: {}): ApiFormFieldSet {
|
|||||||
|
|
||||||
return fields;
|
return fields;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function partParameterTemplateFields(): ApiFormFieldSet {
|
|
||||||
return {
|
|
||||||
name: {},
|
|
||||||
description: {},
|
|
||||||
units: {},
|
|
||||||
choices: {},
|
|
||||||
checkbox: {}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
export function partTestTemplateFields(): ApiFormFieldSet {
|
|
||||||
return {
|
|
||||||
part: {
|
|
||||||
hidden: true
|
|
||||||
},
|
|
||||||
test_name: {},
|
|
||||||
description: {},
|
|
||||||
required: {},
|
|
||||||
requires_value: {},
|
|
||||||
requires_attachment: {}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
@ -1,46 +1,42 @@
|
|||||||
import {
|
import {
|
||||||
|
IconAddressBook,
|
||||||
IconCalendar,
|
IconCalendar,
|
||||||
IconCoins,
|
IconCoins,
|
||||||
IconCurrencyDollar,
|
IconCurrencyDollar,
|
||||||
|
IconHash,
|
||||||
IconLink,
|
IconLink,
|
||||||
|
IconList,
|
||||||
IconNotes,
|
IconNotes,
|
||||||
IconSitemap
|
IconSitemap,
|
||||||
|
IconUser,
|
||||||
|
IconUsers
|
||||||
} from '@tabler/icons-react';
|
} from '@tabler/icons-react';
|
||||||
|
|
||||||
import { ApiFormFieldSet } from '../components/forms/fields/ApiFormField';
|
import {
|
||||||
|
ApiFormAdjustFilterType,
|
||||||
|
ApiFormFieldSet
|
||||||
|
} from '../components/forms/fields/ApiFormField';
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Construct a set of fields for creating / editing a PurchaseOrderLineItem instance
|
* Construct a set of fields for creating / editing a PurchaseOrderLineItem instance
|
||||||
*/
|
*/
|
||||||
export function purchaseOrderLineItemFields({
|
export function purchaseOrderLineItemFields() {
|
||||||
supplierId,
|
|
||||||
orderId,
|
|
||||||
create = false
|
|
||||||
}: {
|
|
||||||
supplierId?: number;
|
|
||||||
orderId?: number;
|
|
||||||
create?: boolean;
|
|
||||||
}) {
|
|
||||||
let fields: ApiFormFieldSet = {
|
let fields: ApiFormFieldSet = {
|
||||||
order: {
|
order: {
|
||||||
filters: {
|
filters: {
|
||||||
supplier_detail: true
|
supplier_detail: true
|
||||||
},
|
},
|
||||||
value: orderId,
|
hidden: true
|
||||||
hidden: create != true || orderId != undefined
|
|
||||||
},
|
},
|
||||||
part: {
|
part: {
|
||||||
filters: {
|
filters: {
|
||||||
part_detail: true,
|
part_detail: true,
|
||||||
supplier_detail: true,
|
supplier_detail: true
|
||||||
supplier: supplierId
|
|
||||||
},
|
},
|
||||||
adjustFilters: (filters: any) => {
|
adjustFilters: (value: ApiFormAdjustFilterType) => {
|
||||||
// TODO: Filter by the supplier associated with the order
|
// TODO: Adjust part based on the supplier associated with the supplier
|
||||||
return filters;
|
return value.filters;
|
||||||
}
|
}
|
||||||
// TODO: Custom onEdit callback (see purchase_order.js)
|
|
||||||
// TODO: secondary modal (see purchase_order.js)
|
|
||||||
},
|
},
|
||||||
quantity: {},
|
quantity: {},
|
||||||
reference: {},
|
reference: {},
|
||||||
@ -66,3 +62,52 @@ export function purchaseOrderLineItemFields({
|
|||||||
|
|
||||||
return fields;
|
return fields;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Construct a set of fields for creating / editing a PurchaseOrder instance
|
||||||
|
*/
|
||||||
|
export function purchaseOrderFields(): ApiFormFieldSet {
|
||||||
|
return {
|
||||||
|
reference: {
|
||||||
|
icon: <IconHash />
|
||||||
|
},
|
||||||
|
description: {},
|
||||||
|
supplier: {
|
||||||
|
filters: {
|
||||||
|
is_supplier: true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
supplier_reference: {},
|
||||||
|
project_code: {
|
||||||
|
icon: <IconList />
|
||||||
|
},
|
||||||
|
order_currency: {
|
||||||
|
icon: <IconCoins />
|
||||||
|
},
|
||||||
|
target_date: {
|
||||||
|
icon: <IconCalendar />
|
||||||
|
},
|
||||||
|
link: {},
|
||||||
|
contact: {
|
||||||
|
icon: <IconUser />,
|
||||||
|
adjustFilters: (value: ApiFormAdjustFilterType) => {
|
||||||
|
return {
|
||||||
|
...value.filters,
|
||||||
|
company: value.data.supplier
|
||||||
|
};
|
||||||
|
}
|
||||||
|
},
|
||||||
|
address: {
|
||||||
|
icon: <IconAddressBook />,
|
||||||
|
adjustFilters: (value: ApiFormAdjustFilterType) => {
|
||||||
|
return {
|
||||||
|
...value.filters,
|
||||||
|
company: value.data.supplier
|
||||||
|
};
|
||||||
|
}
|
||||||
|
},
|
||||||
|
responsible: {
|
||||||
|
icon: <IconUsers />
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
44
src/frontend/src/forms/SalesOrderForms.tsx
Normal file
44
src/frontend/src/forms/SalesOrderForms.tsx
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
import { IconAddressBook, IconUser, IconUsers } from '@tabler/icons-react';
|
||||||
|
|
||||||
|
import {
|
||||||
|
ApiFormAdjustFilterType,
|
||||||
|
ApiFormFieldSet
|
||||||
|
} from '../components/forms/fields/ApiFormField';
|
||||||
|
|
||||||
|
export function salesOrderFields(): ApiFormFieldSet {
|
||||||
|
return {
|
||||||
|
reference: {},
|
||||||
|
description: {},
|
||||||
|
customer: {
|
||||||
|
filters: {
|
||||||
|
is_customer: true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
customer_reference: {},
|
||||||
|
project_code: {},
|
||||||
|
order_currency: {},
|
||||||
|
target_date: {},
|
||||||
|
link: {},
|
||||||
|
contact: {
|
||||||
|
icon: <IconUser />,
|
||||||
|
adjustFilters: (value: ApiFormAdjustFilterType) => {
|
||||||
|
return {
|
||||||
|
...value.filters,
|
||||||
|
company: value.data.customer
|
||||||
|
};
|
||||||
|
}
|
||||||
|
},
|
||||||
|
address: {
|
||||||
|
icon: <IconAddressBook />,
|
||||||
|
adjustFilters: (value: ApiFormAdjustFilterType) => {
|
||||||
|
return {
|
||||||
|
...value.filters,
|
||||||
|
company: value.data.customer
|
||||||
|
};
|
||||||
|
}
|
||||||
|
},
|
||||||
|
responsible: {
|
||||||
|
icon: <IconUsers />
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
@ -107,7 +107,7 @@ export function useCreateStockItem() {
|
|||||||
return useCreateApiFormModal({
|
return useCreateApiFormModal({
|
||||||
url: ApiEndpoints.stock_item_list,
|
url: ApiEndpoints.stock_item_list,
|
||||||
fields: fields,
|
fields: fields,
|
||||||
title: t`Create Stock Item`
|
title: t`Add Stock Item`
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
import { t } from '@lingui/macro';
|
import { t } from '@lingui/macro';
|
||||||
import { Divider, Stack } from '@mantine/core';
|
import { Alert, Divider, Stack } from '@mantine/core';
|
||||||
import { useId } from '@mantine/hooks';
|
import { useId } from '@mantine/hooks';
|
||||||
import { useEffect, useMemo, useRef } from 'react';
|
import { useEffect, useMemo, useRef } from 'react';
|
||||||
|
|
||||||
@ -82,6 +82,8 @@ export function useCreateApiFormModal(props: ApiFormModalProps) {
|
|||||||
const createProps = useMemo<ApiFormModalProps>(
|
const createProps = useMemo<ApiFormModalProps>(
|
||||||
() => ({
|
() => ({
|
||||||
...props,
|
...props,
|
||||||
|
fetchInitialData: props.fetchInitialData ?? false,
|
||||||
|
successMessage: props.successMessage ?? t`Item Created`,
|
||||||
method: 'POST'
|
method: 'POST'
|
||||||
}),
|
}),
|
||||||
[props]
|
[props]
|
||||||
@ -98,7 +100,8 @@ export function useEditApiFormModal(props: ApiFormModalProps) {
|
|||||||
() => ({
|
() => ({
|
||||||
...props,
|
...props,
|
||||||
fetchInitialData: props.fetchInitialData ?? true,
|
fetchInitialData: props.fetchInitialData ?? true,
|
||||||
method: 'PUT'
|
successMessage: props.successMessage ?? t`Item Updated`,
|
||||||
|
method: 'PATCH'
|
||||||
}),
|
}),
|
||||||
[props]
|
[props]
|
||||||
);
|
);
|
||||||
@ -116,6 +119,12 @@ export function useDeleteApiFormModal(props: ApiFormModalProps) {
|
|||||||
method: 'DELETE',
|
method: 'DELETE',
|
||||||
submitText: t`Delete`,
|
submitText: t`Delete`,
|
||||||
submitColor: 'red',
|
submitColor: 'red',
|
||||||
|
successMessage: props.successMessage ?? t`Item Deleted`,
|
||||||
|
preFormContent: props.preFormContent ?? (
|
||||||
|
<Alert
|
||||||
|
color={'red'}
|
||||||
|
>{t`Are you sure you want to delete this item?`}</Alert>
|
||||||
|
),
|
||||||
fields: {}
|
fields: {}
|
||||||
}),
|
}),
|
||||||
[props]
|
[props]
|
||||||
|
@ -42,6 +42,14 @@ window.INVENTREE_SETTINGS = {
|
|||||||
host: `${window.location.origin}/`,
|
host: `${window.location.origin}/`,
|
||||||
name: 'Current Server'
|
name: 'Current Server'
|
||||||
},
|
},
|
||||||
|
...(IS_DEV
|
||||||
|
? {
|
||||||
|
'mantine-2j5j5j5j5': {
|
||||||
|
host: 'http://localhost:8000',
|
||||||
|
name: 'Localhost'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
: {}),
|
||||||
...(IS_DEV_OR_DEMO
|
...(IS_DEV_OR_DEMO
|
||||||
? {
|
? {
|
||||||
'mantine-u56l5jt85': {
|
'mantine-u56l5jt85': {
|
||||||
@ -51,7 +59,11 @@ window.INVENTREE_SETTINGS = {
|
|||||||
}
|
}
|
||||||
: {})
|
: {})
|
||||||
},
|
},
|
||||||
default_server: IS_DEMO ? 'mantine-u56l5jt85' : 'mantine-cqj63coxn',
|
default_server: IS_DEV
|
||||||
|
? 'mantine-2j5j5j5j5'
|
||||||
|
: IS_DEMO
|
||||||
|
? 'mantine-u56l5jt85'
|
||||||
|
: 'mantine-cqj63coxn',
|
||||||
show_server_selector: IS_DEV_OR_DEMO,
|
show_server_selector: IS_DEV_OR_DEMO,
|
||||||
|
|
||||||
// merge in settings that are already set via django's spa_view or for development
|
// merge in settings that are already set via django's spa_view or for development
|
||||||
|
@ -10,42 +10,54 @@ import { StylishText } from '../../components/items/StylishText';
|
|||||||
import { StatusRenderer } from '../../components/render/StatusRenderer';
|
import { StatusRenderer } from '../../components/render/StatusRenderer';
|
||||||
import { ApiEndpoints } from '../../enums/ApiEndpoints';
|
import { ApiEndpoints } from '../../enums/ApiEndpoints';
|
||||||
import { ModelType } from '../../enums/ModelType';
|
import { ModelType } from '../../enums/ModelType';
|
||||||
import {
|
import { partCategoryFields, partFields } from '../../forms/PartForms';
|
||||||
createPart,
|
|
||||||
editPart,
|
|
||||||
partCategoryFields,
|
|
||||||
partFields
|
|
||||||
} from '../../forms/PartForms';
|
|
||||||
import { useCreateStockItem } from '../../forms/StockForms';
|
import { useCreateStockItem } from '../../forms/StockForms';
|
||||||
import {
|
import {
|
||||||
OpenApiFormProps,
|
useCreateApiFormModal,
|
||||||
openCreateApiForm,
|
useEditApiFormModal
|
||||||
openEditApiForm
|
} from '../../hooks/UseForm';
|
||||||
} from '../../functions/forms';
|
|
||||||
import { useCreateApiFormModal } from '../../hooks/UseForm';
|
|
||||||
|
|
||||||
// Generate some example forms using the modal API forms interface
|
// Generate some example forms using the modal API forms interface
|
||||||
const fields = partCategoryFields({});
|
const fields = partCategoryFields({});
|
||||||
|
|
||||||
function ApiFormsPlayground() {
|
function ApiFormsPlayground() {
|
||||||
const editCategoryForm: OpenApiFormProps = {
|
const editCategory = useEditApiFormModal({
|
||||||
url: ApiEndpoints.category_list,
|
url: ApiEndpoints.category_list,
|
||||||
pk: 2,
|
pk: 2,
|
||||||
title: 'Edit Category',
|
title: 'Edit Category',
|
||||||
fields: fields
|
fields: fields
|
||||||
};
|
});
|
||||||
|
|
||||||
const createAttachmentForm: OpenApiFormProps = {
|
const newPart = useCreateApiFormModal({
|
||||||
|
url: ApiEndpoints.part_list,
|
||||||
|
title: 'Create Part',
|
||||||
|
fields: partFields({}),
|
||||||
|
initialData: {
|
||||||
|
description: 'A part created via the API'
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const editPart = useEditApiFormModal({
|
||||||
|
url: ApiEndpoints.part_list,
|
||||||
|
pk: 1,
|
||||||
|
title: 'Edit Part',
|
||||||
|
fields: partFields({ editing: true })
|
||||||
|
});
|
||||||
|
|
||||||
|
const newAttachment = useCreateApiFormModal({
|
||||||
url: ApiEndpoints.part_attachment_list,
|
url: ApiEndpoints.part_attachment_list,
|
||||||
title: 'Create Attachment',
|
title: 'Create Attachment',
|
||||||
successMessage: 'Attachment uploaded',
|
|
||||||
fields: {
|
fields: {
|
||||||
part: {
|
part: {},
|
||||||
value: 1
|
|
||||||
},
|
|
||||||
attachment: {},
|
attachment: {},
|
||||||
comment: {}
|
comment: {}
|
||||||
}
|
},
|
||||||
};
|
initialData: {
|
||||||
|
part: 1
|
||||||
|
},
|
||||||
|
successMessage: 'Attachment uploaded'
|
||||||
|
});
|
||||||
|
|
||||||
const [active, setActive] = useState(true);
|
const [active, setActive] = useState(true);
|
||||||
const [name, setName] = useState('Hello');
|
const [name, setName] = useState('Hello');
|
||||||
|
|
||||||
@ -73,6 +85,14 @@ function ApiFormsPlayground() {
|
|||||||
url: ApiEndpoints.part_list,
|
url: ApiEndpoints.part_list,
|
||||||
title: 'Create part',
|
title: 'Create part',
|
||||||
fields: partFieldsState,
|
fields: partFieldsState,
|
||||||
|
initialData: {
|
||||||
|
is_template: true,
|
||||||
|
virtual: true,
|
||||||
|
minimum_stock: 10,
|
||||||
|
description: 'An example part description',
|
||||||
|
keywords: 'apple, banana, carrottt',
|
||||||
|
'initial_supplier.sku': 'SKU-123'
|
||||||
|
},
|
||||||
preFormContent: (
|
preFormContent: (
|
||||||
<Button onClick={() => setName('Hello world')}>
|
<Button onClick={() => setName('Hello world')}>
|
||||||
Set name="Hello world"
|
Set name="Hello world"
|
||||||
@ -86,19 +106,20 @@ function ApiFormsPlayground() {
|
|||||||
return (
|
return (
|
||||||
<Stack>
|
<Stack>
|
||||||
<Group>
|
<Group>
|
||||||
<Button onClick={() => createPart()}>Create New Part</Button>
|
<Button onClick={() => newPart.open()}>Create New Part</Button>
|
||||||
<Button onClick={() => editPart({ part_id: 1 })}>Edit Part</Button>
|
{newPart.modal}
|
||||||
|
|
||||||
|
<Button onClick={() => editPart.open()}>Edit Part</Button>
|
||||||
|
{editPart.modal}
|
||||||
|
|
||||||
<Button onClick={() => openCreateStockItem()}>Create Stock Item</Button>
|
<Button onClick={() => openCreateStockItem()}>Create Stock Item</Button>
|
||||||
{createStockItemModal}
|
{createStockItemModal}
|
||||||
|
|
||||||
<Button onClick={() => openEditApiForm(editCategoryForm)}>
|
<Button onClick={() => editCategory.open()}>Edit Category</Button>
|
||||||
Edit Category
|
{editCategory.modal}
|
||||||
</Button>
|
|
||||||
|
|
||||||
<Button onClick={() => openCreateApiForm(createAttachmentForm)}>
|
<Button onClick={() => newAttachment.open()}>Create Attachment</Button>
|
||||||
Create Attachment
|
{newAttachment.modal}
|
||||||
</Button>
|
|
||||||
|
|
||||||
<Button onClick={() => openCreatePart()}>Create Part new Modal</Button>
|
<Button onClick={() => openCreatePart()}>Create Part new Modal</Button>
|
||||||
{createPartModal}
|
{createPartModal}
|
||||||
|
@ -14,12 +14,11 @@ import {
|
|||||||
IconQrcode,
|
IconQrcode,
|
||||||
IconSitemap
|
IconSitemap
|
||||||
} from '@tabler/icons-react';
|
} from '@tabler/icons-react';
|
||||||
import { useCallback, useMemo } from 'react';
|
import { useMemo } from 'react';
|
||||||
import { useParams } from 'react-router-dom';
|
import { useParams } from 'react-router-dom';
|
||||||
|
|
||||||
import {
|
import {
|
||||||
ActionDropdown,
|
ActionDropdown,
|
||||||
DeleteItemAction,
|
|
||||||
DuplicateItemAction,
|
DuplicateItemAction,
|
||||||
EditItemAction,
|
EditItemAction,
|
||||||
LinkBarcodeAction,
|
LinkBarcodeAction,
|
||||||
@ -36,8 +35,9 @@ import { StockItemTable } from '../../components/tables/stock/StockItemTable';
|
|||||||
import { NotesEditor } from '../../components/widgets/MarkdownEditor';
|
import { NotesEditor } from '../../components/widgets/MarkdownEditor';
|
||||||
import { ApiEndpoints } from '../../enums/ApiEndpoints';
|
import { ApiEndpoints } from '../../enums/ApiEndpoints';
|
||||||
import { ModelType } from '../../enums/ModelType';
|
import { ModelType } from '../../enums/ModelType';
|
||||||
|
import { UserRoles } from '../../enums/Roles';
|
||||||
import { buildOrderFields } from '../../forms/BuildForms';
|
import { buildOrderFields } from '../../forms/BuildForms';
|
||||||
import { openEditApiForm } from '../../functions/forms';
|
import { useEditApiFormModal } from '../../hooks/UseForm';
|
||||||
import { useInstance } from '../../hooks/UseInstance';
|
import { useInstance } from '../../hooks/UseInstance';
|
||||||
import { apiUrl } from '../../states/ApiState';
|
import { apiUrl } from '../../states/ApiState';
|
||||||
import { useUserState } from '../../states/UserState';
|
import { useUserState } from '../../states/UserState';
|
||||||
@ -152,12 +152,10 @@ export default function BuildDetail() {
|
|||||||
name: 'child-orders',
|
name: 'child-orders',
|
||||||
label: t`Child Build Orders`,
|
label: t`Child Build Orders`,
|
||||||
icon: <IconSitemap />,
|
icon: <IconSitemap />,
|
||||||
content: (
|
content: build.pk ? (
|
||||||
<BuildOrderTable
|
<BuildOrderTable parentBuildId={build.pk} />
|
||||||
params={{
|
) : (
|
||||||
parent: id
|
<Skeleton />
|
||||||
}}
|
|
||||||
/>
|
|
||||||
)
|
)
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -187,24 +185,15 @@ export default function BuildDetail() {
|
|||||||
];
|
];
|
||||||
}, [build, id]);
|
}, [build, id]);
|
||||||
|
|
||||||
const editBuildOrder = useCallback(() => {
|
const editBuild = useEditApiFormModal({
|
||||||
let fields = buildOrderFields();
|
|
||||||
|
|
||||||
// Cannot edit part field after creation
|
|
||||||
fields['part'].hidden = true;
|
|
||||||
|
|
||||||
build.pk &&
|
|
||||||
openEditApiForm({
|
|
||||||
url: ApiEndpoints.build_order_list,
|
url: ApiEndpoints.build_order_list,
|
||||||
pk: build.pk,
|
pk: build.pk,
|
||||||
title: t`Edit Build Order`,
|
title: t`Edit Build Order`,
|
||||||
fields: fields,
|
fields: buildOrderFields(),
|
||||||
successMessage: t`Build Order updated`,
|
|
||||||
onFormSuccess: () => {
|
onFormSuccess: () => {
|
||||||
refreshInstance();
|
refreshInstance();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}, [build]);
|
|
||||||
|
|
||||||
const buildActions = useMemo(() => {
|
const buildActions = useMemo(() => {
|
||||||
// TODO: Disable certain actions based on user permissions
|
// TODO: Disable certain actions based on user permissions
|
||||||
@ -241,10 +230,10 @@ export default function BuildDetail() {
|
|||||||
icon={<IconDots />}
|
icon={<IconDots />}
|
||||||
actions={[
|
actions={[
|
||||||
EditItemAction({
|
EditItemAction({
|
||||||
onClick: editBuildOrder
|
onClick: () => editBuild.open(),
|
||||||
|
disabled: !user.hasChangeRole(UserRoles.build)
|
||||||
}),
|
}),
|
||||||
DuplicateItemAction({}),
|
DuplicateItemAction({})
|
||||||
DeleteItemAction({})
|
|
||||||
]}
|
]}
|
||||||
/>
|
/>
|
||||||
];
|
];
|
||||||
@ -263,6 +252,7 @@ export default function BuildDetail() {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
|
{editBuild.modal}
|
||||||
<Stack spacing="xs">
|
<Stack spacing="xs">
|
||||||
<LoadingOverlay visible={instanceQuery.isFetching} />
|
<LoadingOverlay visible={instanceQuery.isFetching} />
|
||||||
<PageDetail
|
<PageDetail
|
||||||
|
@ -1,45 +1,17 @@
|
|||||||
import { t } from '@lingui/macro';
|
import { t } from '@lingui/macro';
|
||||||
import { Button, Stack } from '@mantine/core';
|
import { Stack } from '@mantine/core';
|
||||||
import { useCallback } from 'react';
|
|
||||||
import { useNavigate } from 'react-router-dom';
|
|
||||||
|
|
||||||
import { PageDetail } from '../../components/nav/PageDetail';
|
import { PageDetail } from '../../components/nav/PageDetail';
|
||||||
import { BuildOrderTable } from '../../components/tables/build/BuildOrderTable';
|
import { BuildOrderTable } from '../../components/tables/build/BuildOrderTable';
|
||||||
import { ApiEndpoints } from '../../enums/ApiEndpoints';
|
|
||||||
import { buildOrderFields } from '../../forms/BuildForms';
|
|
||||||
import { openCreateApiForm } from '../../functions/forms';
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Build Order index page
|
* Build Order index page
|
||||||
*/
|
*/
|
||||||
export default function BuildIndex() {
|
export default function BuildIndex() {
|
||||||
const navigate = useNavigate();
|
|
||||||
|
|
||||||
const newBuildOrder = useCallback(() => {
|
|
||||||
openCreateApiForm({
|
|
||||||
url: ApiEndpoints.build_order_list,
|
|
||||||
title: t`Add Build Order`,
|
|
||||||
fields: buildOrderFields(),
|
|
||||||
successMessage: t`Build order created`,
|
|
||||||
onFormSuccess: (data: any) => {
|
|
||||||
if (data.pk) {
|
|
||||||
navigate(`/build/${data.pk}`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Stack>
|
<Stack>
|
||||||
<PageDetail
|
<PageDetail title={t`Build Orders`} actions={[]} />
|
||||||
title={t`Build Orders`}
|
|
||||||
actions={[
|
|
||||||
<Button key="new-build" color="green" onClick={newBuildOrder}>
|
|
||||||
{t`New Build Order`}
|
|
||||||
</Button>
|
|
||||||
]}
|
|
||||||
/>
|
|
||||||
<BuildOrderTable />
|
<BuildOrderTable />
|
||||||
</Stack>
|
</Stack>
|
||||||
</>
|
</>
|
||||||
|
@ -39,7 +39,8 @@ import { StockItemTable } from '../../components/tables/stock/StockItemTable';
|
|||||||
import { NotesEditor } from '../../components/widgets/MarkdownEditor';
|
import { NotesEditor } from '../../components/widgets/MarkdownEditor';
|
||||||
import { ApiEndpoints } from '../../enums/ApiEndpoints';
|
import { ApiEndpoints } from '../../enums/ApiEndpoints';
|
||||||
import { UserRoles } from '../../enums/Roles';
|
import { UserRoles } from '../../enums/Roles';
|
||||||
import { editCompany } from '../../forms/CompanyForms';
|
import { companyFields } from '../../forms/CompanyForms';
|
||||||
|
import { useEditApiFormModal } from '../../hooks/UseForm';
|
||||||
import { useInstance } from '../../hooks/UseInstance';
|
import { useInstance } from '../../hooks/UseInstance';
|
||||||
import { apiUrl } from '../../states/ApiState';
|
import { apiUrl } from '../../states/ApiState';
|
||||||
import { useUserState } from '../../states/UserState';
|
import { useUserState } from '../../states/UserState';
|
||||||
@ -98,9 +99,7 @@ export default function CompanyDetail(props: CompanyDetailProps) {
|
|||||||
label: t`Purchase Orders`,
|
label: t`Purchase Orders`,
|
||||||
icon: <IconShoppingCart />,
|
icon: <IconShoppingCart />,
|
||||||
hidden: !company?.is_supplier,
|
hidden: !company?.is_supplier,
|
||||||
content: company?.pk && (
|
content: company?.pk && <PurchaseOrderTable supplierId={company.pk} />
|
||||||
<PurchaseOrderTable params={{ supplier: company.pk }} />
|
|
||||||
)
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: 'stock-items',
|
name: 'stock-items',
|
||||||
@ -116,9 +115,7 @@ export default function CompanyDetail(props: CompanyDetailProps) {
|
|||||||
label: t`Sales Orders`,
|
label: t`Sales Orders`,
|
||||||
icon: <IconTruckDelivery />,
|
icon: <IconTruckDelivery />,
|
||||||
hidden: !company?.is_customer,
|
hidden: !company?.is_customer,
|
||||||
content: company?.pk && (
|
content: company?.pk && <SalesOrderTable customerId={company.pk} />
|
||||||
<SalesOrderTable params={{ customer: company.pk }} />
|
|
||||||
)
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: 'return-orders',
|
name: 'return-orders',
|
||||||
@ -179,6 +176,14 @@ export default function CompanyDetail(props: CompanyDetailProps) {
|
|||||||
];
|
];
|
||||||
}, [id, company]);
|
}, [id, company]);
|
||||||
|
|
||||||
|
const editCompany = useEditApiFormModal({
|
||||||
|
url: ApiEndpoints.company_list,
|
||||||
|
pk: company?.pk,
|
||||||
|
title: t`Edit Company`,
|
||||||
|
fields: companyFields(),
|
||||||
|
onFormSuccess: refreshInstance
|
||||||
|
});
|
||||||
|
|
||||||
const companyActions = useMemo(() => {
|
const companyActions = useMemo(() => {
|
||||||
return [
|
return [
|
||||||
<ActionDropdown
|
<ActionDropdown
|
||||||
@ -188,14 +193,7 @@ export default function CompanyDetail(props: CompanyDetailProps) {
|
|||||||
actions={[
|
actions={[
|
||||||
EditItemAction({
|
EditItemAction({
|
||||||
disabled: !user.hasChangeRole(UserRoles.purchase_order),
|
disabled: !user.hasChangeRole(UserRoles.purchase_order),
|
||||||
onClick: () => {
|
onClick: () => editCompany.open()
|
||||||
if (company?.pk) {
|
|
||||||
editCompany({
|
|
||||||
pk: company?.pk,
|
|
||||||
callback: refreshInstance
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}),
|
}),
|
||||||
DeleteItemAction({
|
DeleteItemAction({
|
||||||
disabled: !user.hasDeleteRole(UserRoles.purchase_order)
|
disabled: !user.hasDeleteRole(UserRoles.purchase_order)
|
||||||
@ -206,6 +204,8 @@ export default function CompanyDetail(props: CompanyDetailProps) {
|
|||||||
}, [id, company, user]);
|
}, [id, company, user]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
<>
|
||||||
|
{editCompany.modal}
|
||||||
<Stack spacing="xs">
|
<Stack spacing="xs">
|
||||||
<LoadingOverlay visible={instanceQuery.isFetching} />
|
<LoadingOverlay visible={instanceQuery.isFetching} />
|
||||||
<PageDetail
|
<PageDetail
|
||||||
@ -217,5 +217,6 @@ export default function CompanyDetail(props: CompanyDetailProps) {
|
|||||||
/>
|
/>
|
||||||
<PanelGroup pageKey="company" panels={companyPanels} />
|
<PanelGroup pageKey="company" panels={companyPanels} />
|
||||||
</Stack>
|
</Stack>
|
||||||
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -45,7 +45,7 @@ export default function SupplierPartDetail() {
|
|||||||
label: t`Purchase Orders`,
|
label: t`Purchase Orders`,
|
||||||
icon: <IconShoppingCart />,
|
icon: <IconShoppingCart />,
|
||||||
content: supplierPart?.pk ? (
|
content: supplierPart?.pk ? (
|
||||||
<PurchaseOrderTable params={{ supplier_part: supplierPart.pk }} />
|
<PurchaseOrderTable supplierPartId={supplierPart.pk} />
|
||||||
) : (
|
) : (
|
||||||
<Skeleton />
|
<Skeleton />
|
||||||
)
|
)
|
||||||
|
@ -63,7 +63,8 @@ import { NotesEditor } from '../../components/widgets/MarkdownEditor';
|
|||||||
import { formatPriceRange } from '../../defaults/formatters';
|
import { formatPriceRange } from '../../defaults/formatters';
|
||||||
import { ApiEndpoints } from '../../enums/ApiEndpoints';
|
import { ApiEndpoints } from '../../enums/ApiEndpoints';
|
||||||
import { UserRoles } from '../../enums/Roles';
|
import { UserRoles } from '../../enums/Roles';
|
||||||
import { editPart } from '../../forms/PartForms';
|
import { partFields } from '../../forms/PartForms';
|
||||||
|
import { useEditApiFormModal } from '../../hooks/UseForm';
|
||||||
import { useInstance } from '../../hooks/UseInstance';
|
import { useInstance } from '../../hooks/UseInstance';
|
||||||
import { apiUrl } from '../../states/ApiState';
|
import { apiUrl } from '../../states/ApiState';
|
||||||
import { useUserState } from '../../states/UserState';
|
import { useUserState } from '../../states/UserState';
|
||||||
@ -504,14 +505,7 @@ export default function PartDetail() {
|
|||||||
label: t`Build Orders`,
|
label: t`Build Orders`,
|
||||||
icon: <IconTools />,
|
icon: <IconTools />,
|
||||||
hidden: !part.assembly,
|
hidden: !part.assembly,
|
||||||
content: (
|
content: part?.pk ? <BuildOrderTable partId={part.pk} /> : <Skeleton />
|
||||||
<BuildOrderTable
|
|
||||||
params={{
|
|
||||||
part_detail: true,
|
|
||||||
part: part.pk ?? -1
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
)
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: 'used_in',
|
name: 'used_in',
|
||||||
@ -562,15 +556,7 @@ export default function PartDetail() {
|
|||||||
label: t`Sales Orders`,
|
label: t`Sales Orders`,
|
||||||
icon: <IconTruckDelivery />,
|
icon: <IconTruckDelivery />,
|
||||||
hidden: !part.salable,
|
hidden: !part.salable,
|
||||||
content: part.pk ? (
|
content: part.pk ? <SalesOrderTable partId={part.pk} /> : <Skeleton />
|
||||||
<SalesOrderTable
|
|
||||||
params={{
|
|
||||||
part: part.pk ?? -1
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
) : (
|
|
||||||
<Skeleton />
|
|
||||||
)
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: 'scheduling',
|
name: 'scheduling',
|
||||||
@ -647,6 +633,14 @@ export default function PartDetail() {
|
|||||||
);
|
);
|
||||||
}, [part, id]);
|
}, [part, id]);
|
||||||
|
|
||||||
|
const editPart = useEditApiFormModal({
|
||||||
|
url: ApiEndpoints.part_list,
|
||||||
|
pk: part.pk,
|
||||||
|
title: t`Edit Part`,
|
||||||
|
fields: partFields({ editing: true }),
|
||||||
|
onFormSuccess: refreshInstance
|
||||||
|
});
|
||||||
|
|
||||||
const partActions = useMemo(() => {
|
const partActions = useMemo(() => {
|
||||||
// TODO: Disable actions based on user permissions
|
// TODO: Disable actions based on user permissions
|
||||||
return [
|
return [
|
||||||
@ -685,13 +679,8 @@ export default function PartDetail() {
|
|||||||
actions={[
|
actions={[
|
||||||
DuplicateItemAction({}),
|
DuplicateItemAction({}),
|
||||||
EditItemAction({
|
EditItemAction({
|
||||||
onClick: () => {
|
disabled: !user.hasChangeRole(UserRoles.part),
|
||||||
part.pk &&
|
onClick: () => editPart.open()
|
||||||
editPart({
|
|
||||||
part_id: part.pk,
|
|
||||||
callback: refreshInstance
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}),
|
}),
|
||||||
DeleteItemAction({
|
DeleteItemAction({
|
||||||
disabled: part?.active
|
disabled: part?.active
|
||||||
@ -703,6 +692,7 @@ export default function PartDetail() {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
|
{editPart.modal}
|
||||||
<Stack spacing="xs">
|
<Stack spacing="xs">
|
||||||
<LoadingOverlay visible={instanceQuery.isFetching} />
|
<LoadingOverlay visible={instanceQuery.isFetching} />
|
||||||
<PartCategoryTree
|
<PartCategoryTree
|
||||||
|
@ -20,7 +20,6 @@ export default function PurchasingIndex() {
|
|||||||
label: t`Purchase Orders`,
|
label: t`Purchase Orders`,
|
||||||
icon: <IconShoppingCart />,
|
icon: <IconShoppingCart />,
|
||||||
content: <PurchaseOrderTable />
|
content: <PurchaseOrderTable />
|
||||||
// TODO: Add optional "calendar" display here...
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: 'suppliers',
|
name: 'suppliers',
|
||||||
|
@ -62,11 +62,7 @@ export default function SalesOrderDetail() {
|
|||||||
label: t`Build Orders`,
|
label: t`Build Orders`,
|
||||||
icon: <IconTools />,
|
icon: <IconTools />,
|
||||||
content: order?.pk ? (
|
content: order?.pk ? (
|
||||||
<BuildOrderTable
|
<BuildOrderTable salesOrderId={order.pk} />
|
||||||
params={{
|
|
||||||
sales_order: order.pk
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
) : (
|
) : (
|
||||||
<Skeleton />
|
<Skeleton />
|
||||||
)
|
)
|
||||||
|
Loading…
Reference in New Issue
Block a user