django-bom 1.252__py3-none-any.whl → 1.258__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- bom/admin.py +52 -12
- bom/constants.py +7 -0
- bom/csv_headers.py +22 -44
- bom/forms.py +966 -1019
- bom/helpers.py +79 -4
- bom/migrations/0052_remove_partrevision_attribute_and_more.py +584 -0
- bom/models.py +191 -80
- bom/static/bom/css/style.css +19 -0
- bom/static/bom/js/formset-handler.js +65 -0
- bom/templates/bom/edit-part-class.html +98 -11
- bom/templates/bom/edit-quantity-of-measure.html +119 -0
- bom/templates/bom/edit-user-meta.html +58 -26
- bom/templates/bom/part-info.html +10 -1
- bom/templates/bom/part-revision-display.html +22 -159
- bom/templates/bom/settings.html +138 -11
- bom/tests.py +117 -31
- bom/urls.py +6 -0
- bom/views/views.py +179 -42
- {django_bom-1.252.dist-info → django_bom-1.258.dist-info}/METADATA +1 -1
- {django_bom-1.252.dist-info → django_bom-1.258.dist-info}/RECORD +23 -20
- {django_bom-1.252.dist-info → django_bom-1.258.dist-info}/WHEEL +0 -0
- {django_bom-1.252.dist-info → django_bom-1.258.dist-info}/licenses/LICENSE +0 -0
- {django_bom-1.252.dist-info → django_bom-1.258.dist-info}/top_level.txt +0 -0
bom/admin.py
CHANGED
|
@@ -1,3 +1,4 @@
|
|
|
1
|
+
from django.conf import settings
|
|
1
2
|
from django.contrib import admin
|
|
2
3
|
from django.contrib.auth import get_user_model
|
|
3
4
|
from django.contrib.auth.admin import UserAdmin
|
|
@@ -9,9 +10,13 @@ from .models import (
|
|
|
9
10
|
Part,
|
|
10
11
|
PartClass,
|
|
11
12
|
PartRevision,
|
|
13
|
+
PartRevisionProperty,
|
|
14
|
+
PartRevisionPropertyDefinition,
|
|
12
15
|
Seller,
|
|
13
16
|
SellerPart,
|
|
14
17
|
Subpart,
|
|
18
|
+
UnitDefinition,
|
|
19
|
+
QuantityOfMeasure,
|
|
15
20
|
get_organization_model,
|
|
16
21
|
get_user_meta_model
|
|
17
22
|
)
|
|
@@ -86,7 +91,10 @@ class PartClassAdmin(admin.ModelAdmin):
|
|
|
86
91
|
|
|
87
92
|
class PartRevisionAdminInline(admin.TabularInline):
|
|
88
93
|
model = PartRevision
|
|
94
|
+
extra = 0
|
|
89
95
|
raw_id_fields = ('assembly',)
|
|
96
|
+
readonly_fields = ('timestamp',)
|
|
97
|
+
show_change_link = True
|
|
90
98
|
|
|
91
99
|
|
|
92
100
|
class PartAdmin(admin.ModelAdmin):
|
|
@@ -96,8 +104,10 @@ class PartAdmin(admin.ModelAdmin):
|
|
|
96
104
|
'organization',
|
|
97
105
|
'get_full_part_number',
|
|
98
106
|
)
|
|
107
|
+
list_filter = ('organization', 'number_class',)
|
|
99
108
|
raw_id_fields = ('number_class', 'primary_manufacturer_part',)
|
|
100
109
|
inlines = [
|
|
110
|
+
PartRevisionAdminInline,
|
|
101
111
|
ManufacturerPartAdminInline,
|
|
102
112
|
]
|
|
103
113
|
|
|
@@ -108,10 +118,22 @@ class PartAdmin(admin.ModelAdmin):
|
|
|
108
118
|
get_full_part_number.admin_order_field = 'number_class__part_number'
|
|
109
119
|
|
|
110
120
|
|
|
121
|
+
class QuantityOfMeasureAdmin(admin.ModelAdmin):
|
|
122
|
+
list_display = ('name', 'organization')
|
|
123
|
+
list_filter = ('organization',)
|
|
124
|
+
|
|
125
|
+
|
|
126
|
+
class PartRevisionPropertyInline(admin.TabularInline):
|
|
127
|
+
model = PartRevisionProperty
|
|
128
|
+
extra = 1
|
|
129
|
+
raw_id_fields = ('property_definition', 'part_revision', 'unit_definition')
|
|
130
|
+
|
|
131
|
+
|
|
111
132
|
class PartRevisionAdmin(admin.ModelAdmin):
|
|
112
133
|
list_display = ('part', 'revision', 'description', 'get_assembly_size', 'timestamp',)
|
|
113
134
|
raw_id_fields = ('assembly',)
|
|
114
135
|
readonly_fields = ('timestamp',)
|
|
136
|
+
inlines = [PartRevisionPropertyInline]
|
|
115
137
|
|
|
116
138
|
def get_assembly_size(self, obj):
|
|
117
139
|
return None if obj.assembly is None else obj.assembly.subparts.count()
|
|
@@ -140,22 +162,37 @@ class AssemblyAdmin(admin.ModelAdmin):
|
|
|
140
162
|
]
|
|
141
163
|
|
|
142
164
|
|
|
143
|
-
|
|
165
|
+
class UnitDefinitionAdmin(admin.ModelAdmin):
|
|
166
|
+
list_display = ('name', 'symbol', 'quantity_of_measure', 'base_multiplier', 'organization')
|
|
167
|
+
list_filter = ('organization',)
|
|
168
|
+
|
|
169
|
+
|
|
170
|
+
class PartRevisionPropertyDefinitionAdmin(admin.ModelAdmin):
|
|
171
|
+
list_display = ('code', 'name', 'type', 'organization', 'quantity_of_measure')
|
|
172
|
+
list_filter = ('organization',)
|
|
173
|
+
|
|
174
|
+
|
|
175
|
+
if settings.BOM_USER_META_MODEL == 'bom.UserMeta':
|
|
176
|
+
current_user_admin = admin.site._registry.get(User)
|
|
177
|
+
|
|
178
|
+
if current_user_admin:
|
|
179
|
+
user_admin_class = current_user_admin.__class__
|
|
180
|
+
existing_inlines = list(user_admin_class.inlines or [])
|
|
181
|
+
if UserMetaInline not in existing_inlines:
|
|
182
|
+
existing_inlines.append(UserMetaInline)
|
|
183
|
+
user_admin_class.inlines = existing_inlines
|
|
184
|
+
else:
|
|
185
|
+
class BomUserAdmin(UserAdmin):
|
|
186
|
+
inlines = [UserMetaInline]
|
|
187
|
+
|
|
144
188
|
|
|
145
|
-
|
|
146
|
-
admin_class = current_admin.__class__
|
|
147
|
-
inlines = list(admin_class.inlines or [])
|
|
148
|
-
if UserMetaInline not in inlines:
|
|
149
|
-
inlines.append(UserMetaInline)
|
|
150
|
-
admin_class.inlines = inlines
|
|
151
|
-
else:
|
|
152
|
-
class BomUserAdmin(UserAdmin):
|
|
153
|
-
inlines = [UserMetaInline]
|
|
189
|
+
admin.site.register(User, BomUserAdmin)
|
|
154
190
|
|
|
191
|
+
if settings.BOM_ORGANIZATION_MODEL == 'bom.Organization':
|
|
192
|
+
from .models import Organization
|
|
155
193
|
|
|
156
|
-
admin.site.register(
|
|
194
|
+
admin.site.register(Organization, OrganizationAdmin)
|
|
157
195
|
|
|
158
|
-
admin.site.register(Organization, OrganizationAdmin)
|
|
159
196
|
admin.site.register(Seller, SellerAdmin)
|
|
160
197
|
admin.site.register(SellerPart, SellerPartAdmin)
|
|
161
198
|
admin.site.register(ManufacturerPart, ManufacturerPartAdmin)
|
|
@@ -165,3 +202,6 @@ admin.site.register(PartRevision, PartRevisionAdmin)
|
|
|
165
202
|
admin.site.register(Manufacturer, ManufacturerAdmin)
|
|
166
203
|
admin.site.register(Assembly, AssemblyAdmin)
|
|
167
204
|
admin.site.register(Subpart, SubpartAdmin)
|
|
205
|
+
admin.site.register(UnitDefinition, UnitDefinitionAdmin)
|
|
206
|
+
admin.site.register(PartRevisionPropertyDefinition, PartRevisionPropertyDefinitionAdmin)
|
|
207
|
+
admin.site.register(QuantityOfMeasure, QuantityOfMeasureAdmin)
|
bom/constants.py
CHANGED
|
@@ -34,6 +34,13 @@ DATA_SOURCES = (
|
|
|
34
34
|
(DATA_SOURCE_MOUSER, 'mouser'),
|
|
35
35
|
)
|
|
36
36
|
|
|
37
|
+
PART_REVISION_PROPERTY_TYPE_STRING = 'S'
|
|
38
|
+
PART_REVISION_PROPERTY_TYPE_DECIMAL = 'D'
|
|
39
|
+
PART_REVISION_PROPERTY_TYPE_BOOLEAN = 'B'
|
|
40
|
+
PART_REVISION_PROPERTY_TYPES = ((PART_REVISION_PROPERTY_TYPE_STRING, 'String'),
|
|
41
|
+
(PART_REVISION_PROPERTY_TYPE_DECIMAL, 'Number'),
|
|
42
|
+
(PART_REVISION_PROPERTY_TYPE_BOOLEAN, 'Boolean'),)
|
|
43
|
+
|
|
37
44
|
VALUE_UNITS = (
|
|
38
45
|
NO_CHOICE,
|
|
39
46
|
('Ohms', '\u03A9'),
|
bom/csv_headers.py
CHANGED
|
@@ -35,8 +35,11 @@ class CSVHeader:
|
|
|
35
35
|
class CSVHeaders(ABC):
|
|
36
36
|
all_headers_defns = []
|
|
37
37
|
|
|
38
|
+
def __init__(self, *args, **kwargs):
|
|
39
|
+
self.headers_defns = list(self.all_headers_defns)
|
|
40
|
+
|
|
38
41
|
def get_synoynms(self, hdr_name):
|
|
39
|
-
for defn in self.
|
|
42
|
+
for defn in self.headers_defns:
|
|
40
43
|
if hdr_name in defn:
|
|
41
44
|
return defn.synonyms()
|
|
42
45
|
else:
|
|
@@ -53,7 +56,7 @@ class CSVHeaders(ABC):
|
|
|
53
56
|
|
|
54
57
|
# Preserves order of definitions as listed in all_header_defns:
|
|
55
58
|
def get_default_all(self):
|
|
56
|
-
return [d.name for d in self.
|
|
59
|
+
return [d.name for d in self.headers_defns]
|
|
57
60
|
|
|
58
61
|
# Given a list of header names returns the default name for each. The return list
|
|
59
62
|
# matches the order of the input list. If a name is not recognized, then returns
|
|
@@ -151,6 +154,9 @@ class CSVHeaders(ABC):
|
|
|
151
154
|
return True
|
|
152
155
|
raise CSVHeaderError(f'Missing column named {header}')
|
|
153
156
|
|
|
157
|
+
def add_header(self, header):
|
|
158
|
+
self.headers_defns.append(header)
|
|
159
|
+
|
|
154
160
|
|
|
155
161
|
#
|
|
156
162
|
# For each CSV header class, a static data member dictionary uses the key as the default name for the header while the
|
|
@@ -190,52 +196,25 @@ class PartClassesCSVHeaders(CSVHeaders):
|
|
|
190
196
|
]
|
|
191
197
|
|
|
192
198
|
|
|
193
|
-
class
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
CSVHeader('wavelength', name_options=[]),
|
|
204
|
-
CSVHeader('wavelength_units', name_options=[]),
|
|
205
|
-
CSVHeader('memory', name_options=[]),
|
|
206
|
-
CSVHeader('memory_units', name_options=[]),
|
|
207
|
-
CSVHeader('interface', name_options=[]),
|
|
208
|
-
CSVHeader('supply_voltage', name_options=[]),
|
|
209
|
-
CSVHeader('supply_voltage_units', name_options=[]),
|
|
210
|
-
CSVHeader('temperature_rating', name_options=[]),
|
|
211
|
-
CSVHeader('temperature_rating_units', name_options=[]),
|
|
212
|
-
CSVHeader('power_rating', name_options=[]),
|
|
213
|
-
CSVHeader('power_rating_units', name_options=[]),
|
|
214
|
-
CSVHeader('voltage_rating', name_options=[]),
|
|
215
|
-
CSVHeader('voltage_rating_units', name_options=[]),
|
|
216
|
-
CSVHeader('current_rating', name_options=[]),
|
|
217
|
-
CSVHeader('current_rating_units', name_options=[]),
|
|
218
|
-
CSVHeader('material', name_options=[]),
|
|
219
|
-
CSVHeader('color', name_options=[]),
|
|
220
|
-
CSVHeader('finish', name_options=[]),
|
|
221
|
-
CSVHeader('length', name_options=[]),
|
|
222
|
-
CSVHeader('length_units', name_options=[]),
|
|
223
|
-
CSVHeader('width', name_options=[]),
|
|
224
|
-
CSVHeader('width_units', name_options=[]),
|
|
225
|
-
CSVHeader('height', name_options=[]),
|
|
226
|
-
CSVHeader('height_units', name_options=[]),
|
|
227
|
-
CSVHeader('weight', name_options=[]),
|
|
228
|
-
CSVHeader('weight_units', name_options=[]),
|
|
229
|
-
]
|
|
199
|
+
class PartRevisionPropertyCSVHeaders(CSVHeaders):
|
|
200
|
+
def add_dynamic_headers(self, definitions):
|
|
201
|
+
for defn in definitions:
|
|
202
|
+
self.add_header(
|
|
203
|
+
CSVHeader(defn.form_field_name,
|
|
204
|
+
name_options=[defn.name, f'property_{defn.code}', f'property_{defn.name}', 'p']))
|
|
205
|
+
if defn.quantity_of_measure:
|
|
206
|
+
self.add_header(
|
|
207
|
+
CSVHeader(defn.form_unit_field_name, name_options=[f'{defn.name} Units', f'{defn.code}_units']))
|
|
208
|
+
|
|
230
209
|
|
|
210
|
+
class PartsListCSVHeaders(PartRevisionPropertyCSVHeaders):
|
|
231
211
|
all_headers_defns = [
|
|
232
212
|
CSVHeader('description', name_options=['desc', 'desc.', ]),
|
|
233
213
|
CSVHeader('manufacturer_name', name_options=['mfg_name', 'manufacturer_name', 'part_manufacturer', 'mfg', 'manufacturer', 'manufacturer name', ]),
|
|
234
214
|
CSVHeader('manufacturer_part_number', name_options=['mpn', 'mfg_part_number', 'part_manufacturer_part_number', 'mfg part number', 'manufacturer part number']),
|
|
235
215
|
CSVHeader('part_number', name_options=['part number', 'part no', ]),
|
|
236
216
|
CSVHeader('revision', name_options=['rev', 'part_revision', ]),
|
|
237
|
-
|
|
238
|
-
+ SellerPartCSVHeaders.all_headers_defns
|
|
217
|
+
] + SellerPartCSVHeaders.all_headers_defns
|
|
239
218
|
|
|
240
219
|
|
|
241
220
|
class PartsListCSVHeadersSemiIntelligent(PartsListCSVHeaders):
|
|
@@ -246,11 +225,10 @@ class PartsListCSVHeadersSemiIntelligent(PartsListCSVHeaders):
|
|
|
246
225
|
CSVHeader('part_class', name_options=['class', 'part_category']),
|
|
247
226
|
CSVHeader('part_number', name_options=['part number', 'part no', ]),
|
|
248
227
|
CSVHeader('revision', name_options=['rev', 'part_revision', ]),
|
|
249
|
-
|
|
250
|
-
+ SellerPartCSVHeaders.all_headers_defns
|
|
228
|
+
] + SellerPartCSVHeaders.all_headers_defns
|
|
251
229
|
|
|
252
230
|
|
|
253
|
-
class BOMFlatCSVHeaders(
|
|
231
|
+
class BOMFlatCSVHeaders(PartRevisionPropertyCSVHeaders):
|
|
254
232
|
all_headers_defns = [
|
|
255
233
|
CSVHeader('part_number', name_options=['part number', 'part no', ]),
|
|
256
234
|
CSVHeader('quantity', name_options=['count', 'qty', 'quantity', ]),
|