2025-01-19 22:22:03 -05:00
|
|
|
from rest_framework import viewsets, status
|
|
|
|
from rest_framework.decorators import action
|
|
|
|
from rest_framework.permissions import IsAuthenticated
|
|
|
|
from rest_framework.response import Response
|
2025-07-12 09:20:23 -04:00
|
|
|
from django.contrib.contenttypes.models import ContentType
|
2025-07-10 12:12:03 -04:00
|
|
|
from adventures.models import Location, ContentAttachment
|
2025-01-19 22:22:03 -05:00
|
|
|
from adventures.serializers import AttachmentSerializer
|
2025-07-12 09:20:23 -04:00
|
|
|
from adventures.permissions import IsOwnerOrSharedWithFullAccess
|
|
|
|
|
2025-01-19 22:22:03 -05:00
|
|
|
|
|
|
|
class AttachmentViewSet(viewsets.ModelViewSet):
|
|
|
|
serializer_class = AttachmentSerializer
|
2025-07-12 09:20:23 -04:00
|
|
|
permission_classes = [IsOwnerOrSharedWithFullAccess]
|
2025-01-19 22:22:03 -05:00
|
|
|
|
|
|
|
def get_queryset(self):
|
2025-07-10 12:12:03 -04:00
|
|
|
return ContentAttachment.objects.filter(user=self.request.user)
|
2025-07-12 09:20:23 -04:00
|
|
|
|
2025-01-19 22:22:03 -05:00
|
|
|
def create(self, request, *args, **kwargs):
|
|
|
|
if not request.user.is_authenticated:
|
|
|
|
return Response({"error": "User is not authenticated"}, status=status.HTTP_401_UNAUTHORIZED)
|
2025-07-12 09:20:23 -04:00
|
|
|
|
|
|
|
# Get the content object details
|
|
|
|
content_type_id = request.data.get('content_type')
|
|
|
|
object_id = request.data.get('object_id')
|
|
|
|
|
|
|
|
# For backward compatibility, also check for 'location' parameter
|
Rename Adventures to Locations (#696)
* Refactor user_id to user in adventures and related models, views, and components
- Updated all instances of user_id to user in the adventures app, including models, serializers, views, and frontend components.
- Adjusted queries and filters to reflect the new user field naming convention.
- Ensured consistency across the codebase for user identification in adventures, collections, notes, and transportation entities.
- Modified frontend components to align with the updated data structure, ensuring proper access control and rendering based on user ownership.
* Refactor adventure-related views and components to use "Location" terminology
- Updated GlobalSearchView to replace AdventureSerializer with LocationSerializer.
- Modified IcsCalendarGeneratorViewSet to use LocationSerializer instead of AdventureSerializer.
- Created new LocationImageViewSet for managing location images, including primary image toggling and image deletion.
- Introduced LocationViewSet for managing locations with enhanced filtering, sorting, and sharing capabilities.
- Updated ReverseGeocodeViewSet to utilize LocationSerializer.
- Added ActivityTypesView to retrieve distinct activity types from locations.
- Refactored user views to replace AdventureSerializer with LocationSerializer.
- Updated frontend components to reflect changes from "adventure" to "location", including AdventureCard, AdventureLink, AdventureModal, and others.
- Adjusted API endpoints in frontend routes to align with new location-based structure.
- Ensured all references to adventures are replaced with locations across the codebase.
* refactor: rename adventures to locations across the application
- Updated localization files to replace adventure-related terms with location-related terms.
- Refactored TypeScript types and variables from Adventure to Location in various routes and components.
- Adjusted UI elements and labels to reflect the change from adventures to locations.
- Ensured all references to adventures in the codebase are consistent with the new location terminology.
* Refactor code structure for improved readability and maintainability
* feat: Implement location details page with server-side loading and deletion functionality
- Added +page.server.ts to handle server-side loading of additional location info.
- Created +page.svelte for displaying location details, including images, visits, and maps.
- Integrated GPX file handling and rendering on the map.
- Updated map route to link to locations instead of adventures.
- Refactored profile and search routes to use LocationCard instead of AdventureCard.
* docs: Update terminology from "Adventure" to "Location" and enhance project overview
* docs: Clarify collection examples in usage documentation
* feat: Enable credentials for GPX file fetch and add CORS_ALLOW_CREDENTIALS setting
* Refactor adventure references to locations across the backend and frontend
- Updated CategoryViewSet to reflect location context instead of adventures.
- Modified ChecklistViewSet to include locations in retrieval logic.
- Changed GlobalSearchView to search for locations instead of adventures.
- Adjusted IcsCalendarGeneratorViewSet to handle locations instead of adventures.
- Refactored LocationImageViewSet to remove unused import.
- Updated LocationViewSet to clarify public access for locations.
- Changed LodgingViewSet to reference locations instead of adventures.
- Modified NoteViewSet to prevent listing all locations.
- Updated RecommendationsViewSet to handle locations in parsing and response.
- Adjusted ReverseGeocodeViewSet to search through user locations.
- Updated StatsViewSet to count locations instead of adventures.
- Changed TagsView to reflect activity types for locations.
- Updated TransportationViewSet to reference locations instead of adventures.
- Added new translations for search results related to locations in multiple languages.
- Updated dashboard and profile pages to reflect location counts instead of adventure counts.
- Adjusted search routes to handle locations instead of adventures.
* Update banner image
* style: Update stats component background and border for improved visibility
* refactor: Rename AdventureCard and AdventureModal to LocationCard and LocationModal for consistency
2025-06-25 11:49:34 -04:00
|
|
|
location_id = request.data.get('location')
|
2025-01-19 22:22:03 -05:00
|
|
|
|
2025-07-12 09:20:23 -04:00
|
|
|
if location_id and not (content_type_id and object_id):
|
|
|
|
# Handle legacy location-specific requests
|
|
|
|
try:
|
|
|
|
location = Location.objects.get(id=location_id)
|
|
|
|
content_type = ContentType.objects.get_for_model(Location)
|
|
|
|
content_type_id = content_type.id
|
|
|
|
object_id = location_id
|
|
|
|
except Location.DoesNotExist:
|
|
|
|
return Response({"error": "Location not found"}, status=status.HTTP_404_NOT_FOUND)
|
|
|
|
|
|
|
|
if not (content_type_id and object_id):
|
|
|
|
return Response({"error": "content_type and object_id are required"}, status=status.HTTP_400_BAD_REQUEST)
|
|
|
|
|
|
|
|
try:
|
|
|
|
content_type = ContentType.objects.get(id=content_type_id)
|
|
|
|
model_class = content_type.model_class()
|
|
|
|
content_object = model_class.objects.get(id=object_id)
|
|
|
|
except (ContentType.DoesNotExist, model_class.DoesNotExist):
|
|
|
|
return Response({"error": "Content object not found"}, status=status.HTTP_404_NOT_FOUND)
|
|
|
|
|
2025-01-22 08:36:02 -05:00
|
|
|
return super().create(request, *args, **kwargs)
|
2025-07-12 09:20:23 -04:00
|
|
|
|
2025-01-22 08:36:02 -05:00
|
|
|
def perform_create(self, serializer):
|
2025-07-12 09:20:23 -04:00
|
|
|
content_type_id = self.request.data.get('content_type')
|
|
|
|
object_id = self.request.data.get('object_id')
|
|
|
|
|
|
|
|
# Handle legacy location parameter
|
Rename Adventures to Locations (#696)
* Refactor user_id to user in adventures and related models, views, and components
- Updated all instances of user_id to user in the adventures app, including models, serializers, views, and frontend components.
- Adjusted queries and filters to reflect the new user field naming convention.
- Ensured consistency across the codebase for user identification in adventures, collections, notes, and transportation entities.
- Modified frontend components to align with the updated data structure, ensuring proper access control and rendering based on user ownership.
* Refactor adventure-related views and components to use "Location" terminology
- Updated GlobalSearchView to replace AdventureSerializer with LocationSerializer.
- Modified IcsCalendarGeneratorViewSet to use LocationSerializer instead of AdventureSerializer.
- Created new LocationImageViewSet for managing location images, including primary image toggling and image deletion.
- Introduced LocationViewSet for managing locations with enhanced filtering, sorting, and sharing capabilities.
- Updated ReverseGeocodeViewSet to utilize LocationSerializer.
- Added ActivityTypesView to retrieve distinct activity types from locations.
- Refactored user views to replace AdventureSerializer with LocationSerializer.
- Updated frontend components to reflect changes from "adventure" to "location", including AdventureCard, AdventureLink, AdventureModal, and others.
- Adjusted API endpoints in frontend routes to align with new location-based structure.
- Ensured all references to adventures are replaced with locations across the codebase.
* refactor: rename adventures to locations across the application
- Updated localization files to replace adventure-related terms with location-related terms.
- Refactored TypeScript types and variables from Adventure to Location in various routes and components.
- Adjusted UI elements and labels to reflect the change from adventures to locations.
- Ensured all references to adventures in the codebase are consistent with the new location terminology.
* Refactor code structure for improved readability and maintainability
* feat: Implement location details page with server-side loading and deletion functionality
- Added +page.server.ts to handle server-side loading of additional location info.
- Created +page.svelte for displaying location details, including images, visits, and maps.
- Integrated GPX file handling and rendering on the map.
- Updated map route to link to locations instead of adventures.
- Refactored profile and search routes to use LocationCard instead of AdventureCard.
* docs: Update terminology from "Adventure" to "Location" and enhance project overview
* docs: Clarify collection examples in usage documentation
* feat: Enable credentials for GPX file fetch and add CORS_ALLOW_CREDENTIALS setting
* Refactor adventure references to locations across the backend and frontend
- Updated CategoryViewSet to reflect location context instead of adventures.
- Modified ChecklistViewSet to include locations in retrieval logic.
- Changed GlobalSearchView to search for locations instead of adventures.
- Adjusted IcsCalendarGeneratorViewSet to handle locations instead of adventures.
- Refactored LocationImageViewSet to remove unused import.
- Updated LocationViewSet to clarify public access for locations.
- Changed LodgingViewSet to reference locations instead of adventures.
- Modified NoteViewSet to prevent listing all locations.
- Updated RecommendationsViewSet to handle locations in parsing and response.
- Adjusted ReverseGeocodeViewSet to search through user locations.
- Updated StatsViewSet to count locations instead of adventures.
- Changed TagsView to reflect activity types for locations.
- Updated TransportationViewSet to reference locations instead of adventures.
- Added new translations for search results related to locations in multiple languages.
- Updated dashboard and profile pages to reflect location counts instead of adventure counts.
- Adjusted search routes to handle locations instead of adventures.
* Update banner image
* style: Update stats component background and border for improved visibility
* refactor: Rename AdventureCard and AdventureModal to LocationCard and LocationModal for consistency
2025-06-25 11:49:34 -04:00
|
|
|
location_id = self.request.data.get('location')
|
2025-07-12 09:20:23 -04:00
|
|
|
if location_id and not (content_type_id and object_id):
|
|
|
|
content_type = ContentType.objects.get_for_model(Location)
|
|
|
|
content_type_id = content_type.id
|
|
|
|
object_id = location_id
|
|
|
|
|
|
|
|
content_type = ContentType.objects.get(id=content_type_id)
|
|
|
|
model_class = content_type.model_class()
|
|
|
|
content_object = model_class.objects.get(id=object_id)
|
|
|
|
|
|
|
|
# Determine the appropriate user to assign
|
|
|
|
attachment_user = self._get_attachment_user(content_object)
|
|
|
|
|
|
|
|
serializer.save(
|
|
|
|
user=attachment_user,
|
|
|
|
content_type=content_type,
|
|
|
|
object_id=object_id
|
|
|
|
)
|
|
|
|
|
|
|
|
def _get_attachment_user(self, content_object):
|
|
|
|
"""
|
|
|
|
Determine which user should own the attachment based on the content object.
|
|
|
|
This preserves the original logic for shared collections.
|
|
|
|
"""
|
|
|
|
# Handle Location objects
|
|
|
|
if isinstance(content_object, Location):
|
|
|
|
if content_object.collections.exists():
|
|
|
|
# Get the first collection's owner (assuming all collections have the same owner)
|
|
|
|
collection = content_object.collections.first()
|
|
|
|
return collection.user
|
|
|
|
else:
|
|
|
|
return self.request.user
|
|
|
|
|
|
|
|
# Handle other content types with collections
|
|
|
|
elif hasattr(content_object, 'collection') and content_object.collection:
|
|
|
|
return content_object.collection.user
|
|
|
|
|
|
|
|
# Handle content objects with a user field
|
|
|
|
elif hasattr(content_object, 'user'):
|
|
|
|
return content_object.user
|
|
|
|
|
|
|
|
# Default to request user
|
|
|
|
return self.request.user
|