diff --git a/.github/workflows/starter.yaml b/.github/workflows/starter.yaml index 1c1bd5f72..4594cfe0c 100644 --- a/.github/workflows/starter.yaml +++ b/.github/workflows/starter.yaml @@ -40,7 +40,7 @@ jobs: else echo "environment=review/${{ github.ref_name }}" echo "environment=review/${{ github.ref_name }}" >> $GITHUB_OUTPUT - echo "environment_short=$(echo -n ${{ github.ref_name }} | sed s/feature_// | tr '_' '-' | tr '[:upper:]' '[:lower:]' )" >> $GITHUB_OUTPUT + echo "environment_short=$(echo -n ${{ github.ref_name }} | sed 's/feat\(ure\)\{0,1\}[_/]//' | tr '_' '-' | tr '[:upper:]' '[:lower:]' | cut -c -63 )" >> $GITHUB_OUTPUT fi build_openatlas: needs: [setup_workflow_env] diff --git a/openatlas/api/endpoints/endpoint.py b/openatlas/api/endpoints/endpoint.py index 41bbfd58a..c925b7ebc 100644 --- a/openatlas/api/endpoints/endpoint.py +++ b/openatlas/api/endpoints/endpoint.py @@ -11,15 +11,14 @@ from openatlas import app from openatlas.api.endpoints.parser import Parser from openatlas.api.formats.csv import ( - build_dataframe, build_link_dataframe) + build_dataframe_with_relations, build_dataframe, build_link_dataframe) from openatlas.api.formats.loud import get_loud_entities -from openatlas.api.resources.api_entity import ApiEntity from openatlas.api.resources.resolve_endpoints import ( download, parse_loud_context) from openatlas.api.resources.templates import ( - geojson_collection_template, linked_places_template, loud_template) -from openatlas.api.resources.util import ( - get_linked_entities_api, get_location_link, remove_duplicate_entities) + geojson_collection_template, geojson_pagination, linked_place_pagination, + linked_places_template, loud_pagination, loud_template) +from openatlas.api.resources.util import get_location_link from openatlas.models.entity import Entity, Link @@ -27,9 +26,58 @@ class Endpoint: def __init__( self, entities: Entity | list[Entity], - parser: dict[str, Any]) -> None: + parser: dict[str, Any], + single: bool = False) -> None: self.entities = entities if isinstance(entities, list) else [entities] self.parser = Parser(parser) + self.pagination: dict[str, Any] = {} + self.single = single + self.entities_with_links: dict[int, dict[str, Any]] = {} + self.formated_entities: list[dict[str, Any]] = [] + + def get_links_for_entities(self) -> None: + if not self.entities: + return + for entity in self.entities: + self.entities_with_links[entity.id] = { + 'entity': entity, + 'links': [], + 'links_inverse': []} + for link_ in self.link_parser_check(): + self.entities_with_links[link_.domain.id]['links'].append(link_) + for link_ in self.link_parser_check(inverse=True): + self.entities_with_links[ + link_.range.id]['links_inverse'].append(link_) + + def get_pagination(self) -> None: + total = [e.id for e in self.entities] + count = len(total) + self.parser.limit = self.parser.limit or count + # List of start ids for the index/pages + e_list = [] + if total: + e_list = list(itertools.islice(total, 0, None, self.parser.limit)) + # Creating index + index = \ + [{'page': i + 1, 'startId': id_} for i, id_ in enumerate(e_list)] + if self.parser.page: + self.parser.first = self.parser.get_by_page(index) + # Get which entity should be displayed (first or last) + if self.parser.last or self.parser.first: + total = self.parser.get_start_entity(total) + # Finding position in list of first entity + entity_list_index = 0 + for index_, entity in enumerate(self.entities): + if entity.id == total[0]: + entity_list_index = index_ + break + self.pagination = { + 'count': count, 'index': index, 'entity_index': entity_list_index} + + def reduce_entities_to_limit(self) -> None: + start_index = self.pagination['entity_index'] + end_index = start_index + int(self.parser.limit) + self.entities = self.entities[start_index:end_index] def resolve_entities(self) -> Response | dict[str, Any]: if self.parser.type_id: @@ -37,58 +85,42 @@ def resolve_entities(self) -> Response | dict[str, Any]: if self.parser.search: self.entities = [ e for e in self.entities if self.parser.search_filter(e)] + self.remove_duplicates() + if self.parser.count == 'true': + return jsonify(len(self.entities)) + + self.sort_entities() + self.get_pagination() + self.reduce_entities_to_limit() + self.get_links_for_entities() if self.parser.export == 'csv': return self.export_entities_csv() if self.parser.export == 'csvNetwork': return self.export_csv_for_network_analysis() - self.remove_duplicate_entities() - self.sorting() - result = self.get_json_output() + self.get_entities_formatted() + if self.parser.format in app.config['RDF_FORMATS']: # pragma: no cover return Response( - self.parser.rdf_output(result['results']), + self.parser.rdf_output(self.formated_entities), mimetype=app.config['RDF_FORMATS'][self.parser.format]) - if self.parser.count == 'true': - return jsonify(result['pagination']['entities']) + + result = self.get_json_output() if self.parser.download == 'true': - return download(result, self.parser.get_entities_template()) - return marshal(result, self.parser.get_entities_template()) + return download(result, self.get_entities_template(result)) + return marshal(result, self.get_entities_template(result)) - def resolve_entity(self) -> Response | dict[str, Any] | tuple[Any, int]: - if self.parser.export == 'csv': - return self.export_entities_csv() - if self.parser.export == 'csvNetwork': - return self.export_csv_for_network_analysis() - result = self.get_entity_formatted() - if (self.parser.format - in app.config['RDF_FORMATS']): # pragma: no cover - return Response( - self.parser.rdf_output(result), - mimetype=app.config['RDF_FORMATS'][self.parser.format]) - template = linked_places_template(self.parser) - if self.parser.format in ['geojson', 'geojson-v2']: - template = geojson_collection_template() - if self.parser.format == 'loud': - template = loud_template(result) - if self.parser.download: - return download(result, template) - return marshal(result, template), 200 - - def get_entity_formatted(self) -> dict[str, Any]: - if self.parser.format == 'geojson': - return self.get_geojson() - if self.parser.format == 'geojson-v2': - return self.get_geojson_v2() - entity = self.entities[0] - entity_dict = { - 'entity': entity, - 'links': ApiEntity.get_links_of_entities(entity.id), - 'links_inverse': ApiEntity.get_links_of_entities( - entity.id, inverse=True)} - if self.parser.format == 'loud' \ - or self.parser.format in app.config['RDF_FORMATS']: - return get_loud_entities(entity_dict, parse_loud_context()) - return self.parser.get_linked_places_entity(entity_dict) + def get_json_output(self) -> dict[str, Any]: + if not self.single: + result = { + "results": self.formated_entities, + "pagination": { + 'entitiesPerPage': int(self.parser.limit), + 'entities': self.pagination['count'], + 'index': self.pagination['index'], + 'totalPages': len(self.pagination['index'])}} + else: + result = dict(self.formated_entities[0]) + return result def filter_by_type(self) -> list[Entity]: result = [] @@ -99,25 +131,40 @@ def filter_by_type(self) -> list[Entity]: return result def export_entities_csv(self) -> Response: - frames = [build_dataframe(e, relations=True) for e in self.entities] + frames = [ + build_dataframe_with_relations(e) + for e in self.entities_with_links.values()] return Response( pd.DataFrame(data=frames).to_csv(), mimetype='text/csv', headers={'Content-Disposition': 'attachment;filename=result.csv'}) def export_csv_for_network_analysis(self) -> Response: + entities = [] + links = [] + for items in self.entities_with_links.values(): + entities.append(items['entity']) + for link_ in items['links']: + entities.append(link_.range) + links.append(link_) + for link_inverse in items['links']: + entities.append(link_inverse.domain) + links.append(link_inverse) + self.entities = entities + self.remove_duplicates() + self.get_links_for_entities() archive = BytesIO() with zipfile.ZipFile(archive, 'w') as zipped_file: for key, frame in self.get_entities_grouped_by_class().items(): with zipped_file.open(f'{key}.csv', 'w') as file: - file.write(bytes( - pd.DataFrame(data=frame).to_csv(), encoding='utf8')) + file.write( + bytes( + pd.DataFrame(data=frame).to_csv(), + encoding='utf8')) with zipped_file.open('links.csv', 'w') as file: - link_frame = [ - build_link_dataframe(link_) for link_ in - (self.link_parser_check() - + self.link_parser_check(inverse=True))] - file.write(bytes( + link_frame = [build_link_dataframe(link_) for link_ in links] + file.write( + bytes( pd.DataFrame(data=link_frame).to_csv(), encoding='utf8')) return Response( archive.getvalue(), @@ -125,14 +172,12 @@ def export_csv_for_network_analysis(self) -> Response: headers={'Content-Disposition': 'attachment;filename=oa_csv.zip'}) def get_entities_grouped_by_class(self) -> dict[str, Any]: - self.entities += get_linked_entities_api([e.id for e in self.entities]) - entities = remove_duplicate_entities(self.entities) grouped_entities = {} for class_, entities_ in groupby( - sorted(entities, key=lambda entity: entity.class_.name), + sorted(self.entities, key=lambda entity: entity.class_.name), key=lambda entity: entity.class_.name): grouped_entities[class_] = \ - [build_dataframe(entity) for entity in entities_] + [build_dataframe(entity) for entity in entities_] return grouped_entities def link_parser_check(self, inverse: bool = False) -> list[Link]: @@ -145,7 +190,7 @@ def link_parser_check(self, inverse: bool = False) -> list[Link]: inverse=inverse) return links - def sorting(self) -> None: + def sort_entities(self) -> None: if 'latest' in request.path: return @@ -154,64 +199,37 @@ def sorting(self) -> None: key=self.parser.get_key, reverse=bool(self.parser.sort == 'desc')) - def remove_duplicate_entities(self) -> None: - seen: set[int] = set() - seen_add = seen.add # Faster than always call seen.add() + def remove_duplicates(self) -> None: + exists: set[int] = set() + add_ = exists.add # Faster than always call exists.add() self.entities = \ - [e for e in self.entities if not (e.id in seen or seen_add(e.id))] + [e for e in self.entities if not (e.id in exists or add_(e.id))] - def get_json_output(self) -> dict[str, Any]: - total = [e.id for e in self.entities] - count = len(total) - if self.parser.limit == 0: - self.parser.limit = count - e_list = [] - if total: - e_list = list(itertools.islice(total, 0, None, self.parser.limit)) - index = \ - [{'page': num + 1, 'startId': i} for num, i in enumerate(e_list)] - if index: - self.parser.first = self.parser.get_by_page(index) \ - if self.parser.page else self.parser.first - total = self.parser.get_start_entity(total) \ - if self.parser.last or self.parser.first else total - j = [i for i, x in enumerate(self.entities) if x.id == total[0]] - formatted_entities = [] - if self.entities: - self.entities = [e for idx, e in enumerate(self.entities[j[0]:])] - formatted_entities = self.get_entities_formatted() - return { - "results": formatted_entities, - "pagination": { - 'entitiesPerPage': int(self.parser.limit), - 'entities': count, - 'index': index, - 'totalPages': len(index)}} - - def get_entities_formatted(self) -> list[dict[str, Any]]: - self.entities = self.entities[:int(self.parser.limit)] - if self.parser.format == 'geojson': - return [self.get_geojson()] - if self.parser.format == 'geojson-v2': - return [self.get_geojson_v2()] - entities_dict: dict[int, dict[str, Any]] = {} - for entity in self.entities: - entities_dict[entity.id] = { - 'entity': entity, - 'links': [], - 'links_inverse': []} - for link_ in self.link_parser_check(): - entities_dict[link_.domain.id]['links'].append(link_) - for link_ in self.link_parser_check(inverse=True): - entities_dict[link_.range.id]['links_inverse'].append(link_) - if self.parser.format == 'loud' \ - or self.parser.format in app.config['RDF_FORMATS']: - return [ - get_loud_entities(item, parse_loud_context()) - for item in entities_dict.values()] - return [ - self.parser.get_linked_places_entity(item) - for item in entities_dict.values()] + def get_entities_formatted(self) -> None: + if not self.entities: + return + entities = [] + match self.parser.format: + case 'geojson': + entities = [self.get_geojson()] + case 'geojson-v2': + entities = [self.get_geojson_v2()] + case 'loud': + parsed_context = parse_loud_context() + entities = [ + get_loud_entities(item, parsed_context) + for item in self.entities_with_links.values()] + case 'lp' | 'lpx': + entities = [ + self.parser.get_linked_places_entity(item) + for item in self.entities_with_links.values()] + case _ if self.parser.format \ + in app.config['RDF_FORMATS']: # pragma: no cover + parsed_context = parse_loud_context() + entities = [ + get_loud_entities(item, parsed_context) + for item in self.entities_with_links.values()] + self.formated_entities = entities def get_geojson(self) -> dict[str, Any]: out = [] @@ -247,3 +265,19 @@ def get_geojson_v2(self) -> dict[str, Any]: [l_.range.id for l_ in entity_links]): out.append(self.parser.get_geojson_dict(entity, geom)) return {'type': 'FeatureCollection', 'features': out} + + def get_entities_template(self, result: dict[str, Any]) -> dict[str, Any]: + match self.parser.format: + case 'geojson' | 'geojson-v2': + template = geojson_collection_template() + if not self.single: + template = geojson_pagination() + case 'loud': + template = loud_template(result) + if not self.single: + template = loud_pagination() + case 'lp' | 'lpx' | _: + template = linked_places_template(self.parser) + if not self.single: + template = linked_place_pagination(self.parser) + return template diff --git a/openatlas/api/endpoints/entities.py b/openatlas/api/endpoints/entities.py index 666cb417d..c723d6ba3 100644 --- a/openatlas/api/endpoints/entities.py +++ b/openatlas/api/endpoints/entities.py @@ -20,7 +20,6 @@ def get(class_: str) -> tuple[Resource, int] | Response | dict[str, Any]: ApiEntity.get_by_cidoc_classes([class_]), entity_.parse_args()).resolve_entities() - class GetBySystemClass(Resource): @staticmethod def get(class_: str) -> tuple[Resource, int] | Response | dict[str, Any]: @@ -61,7 +60,8 @@ class GetEntity(Resource): def get(id_: int) -> tuple[Resource, int] | Response | dict[str, Any]: return Endpoint( ApiEntity.get_by_id(id_, types=True, aliases=True), - entity_.parse_args()).resolve_entity() + entity_.parse_args(), + single=True).resolve_entities() class GetLatest(Resource): diff --git a/openatlas/api/endpoints/parser.py b/openatlas/api/endpoints/parser.py index dc425b7aa..453828961 100644 --- a/openatlas/api/endpoints/parser.py +++ b/openatlas/api/endpoints/parser.py @@ -21,8 +21,6 @@ from openatlas.api.resources.search import get_search_values, search_entity from openatlas.api.resources.search_validation import ( check_if_date_search, validate_search_parameters) -from openatlas.api.resources.templates import ( - geojson_pagination, linked_place_pagination, loud_pagination) from openatlas.api.resources.util import ( flatten_list_and_remove_duplicates, get_geometric_collection, get_geoms_dict, get_location_link, get_reference_systems, @@ -289,13 +287,6 @@ def rdf_output( graph = Graph().parse(data=json.dumps(data), format='json-ld') return graph.serialize(format=self.format, encoding='utf-8') - def get_entities_template(self) -> dict[str, Any]: - if self.format in ['geojson', 'geojson-v2']: - return geojson_pagination() - if self.format == 'loud': - return loud_pagination() - return linked_place_pagination(self) - def is_valid_url(self) -> None: if self.url and isinstance( validators.url(self.url), diff --git a/openatlas/api/formats/csv.py b/openatlas/api/formats/csv.py index d9930e47a..b3a0b5357 100644 --- a/openatlas/api/formats/csv.py +++ b/openatlas/api/formats/csv.py @@ -11,11 +11,20 @@ from openatlas.models.gis import Gis -def build_dataframe( - entity: Entity, - relations: bool = False) -> dict[str, Any]: +def build_dataframe_with_relations( + entity_dict: dict[str, Any]) -> dict[str, Any]: + entity = entity_dict['entity'] + data = build_dataframe(entity) + for key, value in get_csv_links(entity_dict).items(): + data[key] = ' | '.join(list(map(str, value))) + for key, value in get_csv_types(entity_dict).items(): + data[key] = ' | '.join(list(map(str, value))) + return data + + +def build_dataframe(entity: Entity) -> dict[str, Any]: geom = get_csv_geom_entry(entity) - data = { + return { 'id': str(entity.id), 'name': entity.name, 'description': entity.description, @@ -29,12 +38,6 @@ def build_dataframe( 'system_class': entity.class_.name, 'geom_type': geom['type'], 'coordinates': geom['coordinates']} - if relations: - for key, value in get_csv_links(entity).items(): - data[key] = ' | '.join(list(map(str, value))) - for key, value in get_csv_types(entity).items(): - data[key] = ' | '.join(list(map(str, value))) - return data def build_link_dataframe(link: Link) -> dict[str, Any]: @@ -53,28 +56,28 @@ def build_link_dataframe(link: Link) -> dict[str, Any]: 'end_comment': link.end_comment} -def get_csv_types(entity: Entity) -> dict[Any, list[Any]]: +def get_csv_types(entity_dict: dict[str, Any]) -> dict[Any, list[Any]]: types: dict[str, Any] = defaultdict(list) - for type_ in entity.types: + for type_ in entity_dict['entity'].types: hierarchy = [g.types[root].name for root in type_.root] value = '' - for link in Entity.get_links_of_entities(entity.id): + for link in entity_dict['links']: if link.range.id == type_.id and link.description: value += link.description if link.range.id == type_.id and type_.description: value += f' {type_.description}' key = ' > '.join(map(str, hierarchy)) - types[key].append(f"{type_.name}: {value or ''}") + types[key].append(type_.name + (f": {value}" if value else '')) return types -def get_csv_links(entity: Entity) -> dict[str, Any]: +def get_csv_links(entity_dict: dict[str, Any]) -> dict[str, Any]: links: dict[str, Any] = defaultdict(list) - for link in Entity.get_links_of_entities(entity.id): + for link in entity_dict['links']: key = f"{link.property.i18n['en'].replace(' ', '_')}_" \ f"{link.range.class_.name}" links[key].append(link.range.name) - for link in Entity.get_links_of_entities(entity.id, inverse=True): + for link in entity_dict['links_inverse']: key = f"{link.property.i18n['en'].replace(' ', '_')}_" \ f"{link.range.class_.name}" if link.property.i18n_inverse['en']: @@ -124,12 +127,12 @@ def export_database_csv(tables: dict[str, Any], filename: str) -> Response: f'{system_class}.csv', 'w') as file: file.write(bytes( pd.DataFrame(data=frame).to_csv(), - encoding='utf8')) + encoding='utf-8')) else: with zipped_file.open(f'{name}.csv', 'w') as file: file.write(bytes( pd.DataFrame(data=entries).to_csv(), - encoding='utf8')) + encoding='utf-8')) return Response( archive.getvalue(), mimetype='application/zip', diff --git a/openatlas/api/formats/loud.py b/openatlas/api/formats/loud.py index 24a799baa..8bac1b2a0 100644 --- a/openatlas/api/formats/loud.py +++ b/openatlas/api/formats/loud.py @@ -8,7 +8,7 @@ from openatlas.api.resources.util import ( remove_spaces_dashes, date_to_str, get_crm_relation, get_crm_code) from openatlas.display.util import get_file_path -from openatlas.models.entity import Entity +from openatlas.models.entity import Entity, Link from openatlas.models.gis import Gis from openatlas.models.type import Type @@ -45,10 +45,8 @@ def get_domain_links() -> dict[str, Any]: for link_ in data['links']: if link_.property.code in ['OA7', 'OA8', 'OA9']: continue - if link_.property.code == 'P127': - property_name = 'broader' - else: - property_name = loud[get_crm_relation(link_).replace(' ', '_')] + property_name = get_loud_property_name(loud, link_) + if link_.property.code == 'P53': for geom in Gis.get_wkt_by_id(link_.range.id): base_property = get_range_links() | geom @@ -61,11 +59,7 @@ def get_domain_links() -> dict[str, Any]: for link_ in data['links_inverse']: if link_.property.code in ['OA7', 'OA8', 'OA9']: continue - if link_.property.code == 'P127': - property_name = 'broader' - else: - property_name = \ - loud[get_crm_relation(link_, True).replace(' ', '_')] + property_name = get_loud_property_name(loud, link_, inverse=True) if link_.property.code == 'P53': for geom in Gis.get_wkt_by_id(link_.range.id): @@ -77,37 +71,53 @@ def get_domain_links() -> dict[str, Any]: if link_.domain.class_.name == 'file' and g.files.get(link_.domain.id): image_links.append(link_) + if image_links: - profile_image = Entity.get_profile_image_id(data['entity']) - representation: dict[str, Any] = { - 'type': 'VisualItem', - 'digitally_shown_by': []} - for link_ in image_links: - id_ = link_.domain.id - mime_type, _ = mimetypes.guess_type(g.files[id_]) - if not mime_type: - continue # pragma: no cover - file_ = get_file_path(id_) - image = { - 'id': url_for('api.entity', id_=id_, _external=True), - '_label': link_.domain.name, - 'type': 'DigitalObject', - 'format': mime_type, - 'access_point': [{ - 'id': url_for( - 'api.display', - filename=file_.stem if file_ else '', - _external=True), - 'type': 'DigitalObject', - '_label': 'ProfileImage' if id_ == profile_image else ''}]} - if type_ := get_standard_type_loud(link_.domain.types): - image['classified_as'] = get_type_property(type_) - representation['digitally_shown_by'].append(image) - properties_set['representation'].append(representation) + properties_set['representation'].append( + get_loud_images(data['entity'], image_links)) return {'@context': app.config['API_CONTEXT']['LOUD']} | \ base_entity_dict() | properties_set +def get_loud_property_name( + loud: dict[str, str], + link_: Link, + inverse: bool = False) -> str: + name = 'broader' + if not link_.property.code == 'P127': + name = loud[get_crm_relation(link_, inverse).replace(' ', '_')] + return name + + +def get_loud_images(entity: Entity, image_links: list[Link]) -> dict[str, Any]: + profile_image = Entity.get_profile_image_id(entity) + representation: dict[str, Any] = { + 'type': 'VisualItem', + 'digitally_shown_by': []} + for link_ in image_links: + id_ = link_.domain.id + mime_type, _ = mimetypes.guess_type(g.files[id_]) + if not mime_type: + continue # pragma: no cover + file_ = get_file_path(id_) + image = { + 'id': url_for('api.entity', id_=id_, _external=True), + '_label': link_.domain.name, + 'type': 'DigitalObject', + 'format': mime_type, + 'access_point': [{ + 'id': url_for( + 'api.display', + filename=file_.stem if file_ else '', + _external=True), + 'type': 'DigitalObject', + '_label': 'ProfileImage' if id_ == profile_image else ''}]} + if type_ := get_standard_type_loud(link_.domain.types): + image['classified_as'] = get_type_property(type_) + representation['digitally_shown_by'].append(image) + return representation + + def get_loud_timespan(entity: Entity) -> dict[str, Any]: return { 'type': 'TimeSpan', diff --git a/openatlas/api/resources/parser.py b/openatlas/api/resources/parser.py index bc5d461a4..079904936 100644 --- a/openatlas/api/resources/parser.py +++ b/openatlas/api/resources/parser.py @@ -106,6 +106,7 @@ type=str, help='{error_msg}', case_sensitive=False, + default='lp', choices=frozenset(app.config['API_FORMATS']), location='args') entity_.add_argument( diff --git a/openatlas/models/entity.py b/openatlas/models/entity.py index 99d84b09e..8d7eee70a 100644 --- a/openatlas/models/entity.py +++ b/openatlas/models/entity.py @@ -654,6 +654,7 @@ def __init__( self.last = format_date_part(self.end_to, 'year') \ if self.end_to else self.last + def update(self) -> None: db_link.update({ 'id': self.id, diff --git a/openatlas/static/css/style.css b/openatlas/static/css/style.css index 5e3578e70..b897ab2eb 100644 --- a/openatlas/static/css/style.css +++ b/openatlas/static/css/style.css @@ -257,3 +257,12 @@ img.schema_image { min-width: 12.25rem; } +.active > span > .tab-counter.badge { + color: var(--bs-primary); + background-color: white; +} + +:not(.active) > span > .tab-counter.badge { + background-color: var(--bs-primary); + color: white; +} diff --git a/openatlas/templates/entity/view.html b/openatlas/templates/entity/view.html index 52835848f..0d5237187 100644 --- a/openatlas/templates/entity/view.html +++ b/openatlas/templates/entity/view.html @@ -10,7 +10,7 @@

{{ entity.name }}

{% if frontend_link %} - {{ frontend_link|safe }} + {{ frontend_link|safe }} {% endif %}
{{ entity|profile_image|safe }} diff --git a/openatlas/templates/tabs.html b/openatlas/templates/tabs.html index cc6ee01e1..1da3687e9 100644 --- a/openatlas/templates/tabs.html +++ b/openatlas/templates/tabs.html @@ -14,14 +14,15 @@ aria-selected="{% if active %}true{% else %}false{% endif %}" href="#tab-{{ tab.name|replace('_', '-') }}"> - {{ _(tab.name|replace('_', ' '))|uc_first }} - {% if tab.table.rows|length %} - {{ '{0:,}'.format(tab.table.rows|length) }} - {% endif %} + data-bs-toggle="tooltip" + title="{{ tab.tooltip }}" + data-bs-placement="top" + class="d-flex gap-1"> + {{ _(tab.name|replace('_', ' '))|uc_first }} + {% if tab.table.rows|length %} + {{ + '{0:,}'.format(tab.table.rows|length) }} + {% endif %} diff --git a/openatlas/views/changelog.py b/openatlas/views/changelog.py index 0d9362a94..f3fef7061 100644 --- a/openatlas/views/changelog.py +++ b/openatlas/views/changelog.py @@ -15,7 +15,12 @@ def index_changelog() -> str: # pylint: disable=too-many-lines versions = { - '8.10.0': ['TBA', {}], + '8.10.0': ['TBA', { + 'feature': { + '2417': 'Make count from tabs more visible', + '2415': 'Manual: How to report an issue on redmine', + '2444': 'Refactor and minor improvements'} + }], '8.9.0': ['2025-01-01', { 'feature': { '2079': 'Text annotation', diff --git a/tests/test_admin.py b/tests/test_admin.py index 0cf99757a..2af512dc4 100644 --- a/tests/test_admin.py +++ b/tests/test_admin.py @@ -96,7 +96,7 @@ def test_admin(self) -> None: source.link('P2', g.types[source_type.subs[1]]) rv = c.get(url_for('check_dates')) - assert b'' in rv.data + assert b'tab-counter' in rv.data rv = c.get(url_for('check_link_duplicates')) assert b'Event Horizon' in rv.data diff --git a/tests/test_api.py b/tests/test_api.py index 12dbf9019..c4c7233f5 100644 --- a/tests/test_api.py +++ b/tests/test_api.py @@ -277,6 +277,7 @@ def test_api(self) -> None: export='csvNetwork'))]: assert b'Shire' in rv.data assert 'application/zip' in rv.headers.get('Content-Type') + rv = c.get( url_for( 'api_04.linked_entities_by_properties_recursive',