1
|
|
|
# -*- coding: utf-8 -*- |
2
|
|
|
# |
3
|
|
|
# This file is part of SENAITE.CORE. |
4
|
|
|
# |
5
|
|
|
# SENAITE.CORE is free software: you can redistribute it and/or modify it under |
6
|
|
|
# the terms of the GNU General Public License as published by the Free Software |
7
|
|
|
# Foundation, version 2. |
8
|
|
|
# |
9
|
|
|
# This program is distributed in the hope that it will be useful, but WITHOUT |
10
|
|
|
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS |
11
|
|
|
# FOR A PARTICULAR PURPOSE. See the GNU General Public License for more |
12
|
|
|
# details. |
13
|
|
|
# |
14
|
|
|
# You should have received a copy of the GNU General Public License along with |
15
|
|
|
# this program; if not, write to the Free Software Foundation, Inc., 51 |
16
|
|
|
# Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. |
17
|
|
|
# |
18
|
|
|
# Copyright 2018-2025 by it's authors. |
19
|
|
|
# Some rights reserved, see README and LICENSE. |
20
|
|
|
|
21
|
|
|
import glob |
22
|
|
|
import os |
23
|
|
|
|
24
|
|
|
from pkg_resources import resource_filename |
25
|
|
|
from plone.resource.utils import iterDirectoriesOfType |
26
|
|
|
from senaite.core.schema.vocabulary import to_simple_vocabulary |
27
|
|
|
from zope.interface import implementer |
28
|
|
|
from zope.schema.interfaces import IVocabularyFactory |
29
|
|
|
|
30
|
|
|
|
31
|
|
|
@implementer(IVocabularyFactory) |
32
|
|
|
class StickerTemplatesVocabulary(object): |
33
|
|
|
"""Provides all available stickers |
34
|
|
|
""" |
35
|
|
|
|
36
|
|
|
def __call__(self, context, filter_by_type=False): |
37
|
|
|
templates = get_sticker_templates(filter_by_type=filter_by_type) |
38
|
|
|
return to_simple_vocabulary([(t["id"], t["title"]) for t in templates]) |
39
|
|
|
|
40
|
|
|
|
41
|
|
|
StickerTemplatesVocabularyFactory = StickerTemplatesVocabulary() |
42
|
|
|
|
43
|
|
|
|
44
|
|
|
def get_sticker_templates(filter_by_type=None): |
45
|
|
|
"""Returns a list of sticker records |
46
|
|
|
|
47
|
|
|
Each sticker record is a dictionary of the following structure: |
48
|
|
|
|
49
|
|
|
{ |
50
|
|
|
"id": <template_id>, |
51
|
|
|
"title": <template_title>, |
52
|
|
|
} |
53
|
|
|
|
54
|
|
|
If the template lives outside the senaite.core add-on, both the |
55
|
|
|
template_id and template_title include a prefix that matches with |
56
|
|
|
the add-on identifier. |
57
|
|
|
|
58
|
|
|
The template_title is the same name as the id, but with whitespaces and |
59
|
|
|
without extension. |
60
|
|
|
|
61
|
|
|
As an example, for a template from the my.product add-on located in |
62
|
|
|
templates/stickers, and with a filename "EAN128_default_small.pt", |
63
|
|
|
the dictionary will look like: |
64
|
|
|
|
65
|
|
|
{ |
66
|
|
|
"id": "my.product:EAN128_default_small.pt", |
67
|
|
|
"title": "my.product: EAN128 default small", |
68
|
|
|
} |
69
|
|
|
|
70
|
|
|
If filter by type is given in the request, only the templates under the |
71
|
|
|
path with the type name will be fetched. |
72
|
|
|
|
73
|
|
|
Example: If filter_by_type=='worksheet', only *.pt files under a folder |
74
|
|
|
with this name will be displayed. |
75
|
|
|
|
76
|
|
|
:param filter_by_type: sticker type, e.g. "batch" or "worksheet" |
77
|
|
|
:returns: list of sticker records |
78
|
|
|
""" |
79
|
|
|
resdirname = "stickers" |
80
|
|
|
if filter_by_type: |
81
|
|
|
fs_path = os.path.join( |
82
|
|
|
"browser", "stickers", "templates", resdirname, filter_by_type) |
83
|
|
|
else: |
84
|
|
|
fs_path = os.path.join("browser", "stickers", "templates", resdirname) |
85
|
|
|
|
86
|
|
|
templates_dir = resource_filename("senaite.core", fs_path) |
87
|
|
|
templates_subdir = os.path.join(templates_dir, "*.pt") |
88
|
|
|
templates = [os.path.split(x)[-1] for x in glob.glob(templates_subdir)] |
89
|
|
|
|
90
|
|
|
# Retrieve the templates from other add-ons |
91
|
|
|
for templates_resource in iterDirectoriesOfType(resdirname): |
92
|
|
|
prefix = templates_resource.__name__ |
93
|
|
|
if prefix == "senaite.core": |
94
|
|
|
continue |
95
|
|
|
directory = templates_resource.directory |
96
|
|
|
# Only use the directory asked in "filter_by_type" |
97
|
|
|
if filter_by_type: |
98
|
|
|
directory = directory + "/" + filter_by_type |
99
|
|
|
if os.path.isdir(directory): |
100
|
|
|
dirlist = os.listdir(directory) |
101
|
|
|
exts = ["{0}:{1}".format(prefix, tpl) for tpl in dirlist if |
102
|
|
|
tpl.endswith(".pt")] |
103
|
|
|
templates.extend(exts) |
104
|
|
|
|
105
|
|
|
out = [] |
106
|
|
|
templates.sort() |
107
|
|
|
for template in templates: |
108
|
|
|
title = template[:-3] |
109
|
|
|
title = title.replace("_", " ") |
110
|
|
|
title = title.replace(":", ": ") |
111
|
|
|
out.append({"id": template, |
112
|
|
|
"title": title}) |
113
|
|
|
|
114
|
|
|
return out |
115
|
|
|
|