Created initial publish functionality, layout polish

This commit is contained in:
Barunes Padhy 2024-06-08 22:53:43 +03:00
parent 4f92620875
commit 81b0534ad0
10 changed files with 130 additions and 33 deletions

1
backend/.gitignore vendored
View File

@ -10,6 +10,7 @@ local_settings.py
db.sqlite3
db.sqlite3-journal
media
.idea
# If your build process includes running collectstatic, then you probably don't need or want to include staticfiles/
# in your Git repository. Update and uncomment the following line accordingly.

View File

@ -0,0 +1,7 @@
from django.core.files.storage import FileSystemStorage
class CustomStorage(FileSystemStorage):
def __init__(self, location=None, base_url=None):
location = location or 'deploy/data/' # Define your custom path
base_url = base_url or ''
super().__init__(location, base_url)

View File

@ -2,13 +2,15 @@
import os
import shutil
import random
import json
import ast
from rest_framework import generics, status
from rest_framework.views import APIView
from rest_framework.response import Response
from rest_framework.parsers import MultiPartParser, FormParser
from django.core.files.storage import default_storage
from django.core.files.base import ContentFile
from django.conf import settings
from django.http import JsonResponse
#################################################################
#API related imports
from .models import (
@ -26,9 +28,7 @@ from .serializers import (
)
################################################################
#Custom Imports
from .MediaHandler import (
MediaHandler,
)
from .custom_storage import CustomStorage
#UserData related views#############################################
class UserDataUpdateAPIView(generics.RetrieveUpdateAPIView):
@ -114,7 +114,7 @@ class BlogDeleteAPIView(generics.DestroyAPIView):
def remove_directory(self, instance):
print(f"Deleting media files for {instance}")
media_folder = os.path.join(settings.MEDIA_ROOT, 'rangolio_data', 'blog', str(instance.blog_id))
media_folder = os.path.join(settings.MEDIA_ROOT, 'data', 'blog', str(instance.blog_id))
try:
shutil.rmtree(media_folder)
print(f"Directory '{media_folder}' and all its contents have been removed")
@ -133,7 +133,7 @@ class MediaUpload(APIView):
files = request.FILES.getlist('media')
resource_type = file_serializer.validated_data['resource_type']
resource_id = file_serializer.validated_data['resource_id']
file_path_base = f'rangolio_data'
file_path_base = f'data'
for f in files:
file_unique_slug = ''.join(random.choices('ABCDEabcde12345', k=6))
@ -150,26 +150,26 @@ class MediaUpload(APIView):
class ListMedia(APIView):
def get(self, request, resource_type, resource_id, format=None):
if resource_id != resource_type:
media_folder = os.path.join(settings.MEDIA_ROOT, 'rangolio_data', resource_type, resource_id, 'media')
media_folder = os.path.join(settings.MEDIA_ROOT, 'data', resource_type, resource_id, 'media')
else:
media_folder = os.path.join(settings.MEDIA_ROOT, 'rangolio_data', resource_type, 'media')
media_folder = os.path.join(settings.MEDIA_ROOT, 'data', resource_type, 'media')
if not os.path.exists(media_folder):
return Response({'error': 'Media directory not found'}, status=status.HTTP_404_NOT_FOUND)
media_files = [f for f in os.listdir(media_folder) if f.endswith(('.png', '.jpg', '.jpeg'))]
if resource_id != resource_type:
media_urls = [request.build_absolute_uri(f'{settings.MEDIA_URL}rangolio_data/{resource_type}/{resource_id}/media/' + f) for f in media_files]
media_urls = [request.build_absolute_uri(f'{settings.MEDIA_URL}data/{resource_type}/{resource_id}/media/' + f) for f in media_files]
else:
media_urls = [request.build_absolute_uri(f'{settings.MEDIA_URL}rangolio_data/{resource_type}/media/' + f) for f in media_files]
media_urls = [request.build_absolute_uri(f'{settings.MEDIA_URL}data/{resource_type}/media/' + f) for f in media_files]
return Response({'media': media_urls}, status=status.HTTP_200_OK)
def delete(self, request, resource_type, resource_id, format=None):
if resource_id != resource_type:
media_folder = os.path.join(settings.MEDIA_ROOT, 'rangolio_data', resource_type, resource_id, 'media')
media_folder = os.path.join(settings.MEDIA_ROOT, 'data', resource_type, resource_id, 'media')
else:
media_folder = os.path.join(settings.MEDIA_ROOT, 'rangolio_data', resource_type, 'media')
media_folder = os.path.join(settings.MEDIA_ROOT, 'data', resource_type, 'media')
file_name = request.query_params.get('file')
if not file_name or not file_name.endswith(('.png', '.jpg', '.jpeg')):
return Response({'error': 'Invalid or no file name provided'}, status=status.HTTP_400_BAD_REQUEST)
@ -183,10 +183,107 @@ class ListMedia(APIView):
class Publish(APIView):
def get(self, request, deploy_type, format=None):
self.create_json()
storage = CustomStorage()
self.create_json(storage)
return Response({"deploy_type": deploy_type}, status=status.HTTP_200_OK)
def create_json(self):
print(UserData.objects.all())
print(Category.objects.all())
print(Blog.objects.all())
def create_json(self, storage):
self.create_user_data_json(UserData.objects.first(), storage)
self.create_theme_data_json(UserData.objects.first(), storage)
self.create_category_data_json(Category, storage)
self.create_blog_data_json(Blog.objects.all(), storage)
self.merge_media()
def create_user_data_json(self, instance, storage):
json_content = {
"name": instance.name,
"introContent": instance.intro_content,
"profilePhoto": self.sanitize_media_link(instance.profile_photo),
"builtWith": instance.built_with
}
self.save_json(json_content, 'shared/user-data.json', storage)
def create_theme_data_json(self, instance, storage):
json_content = {
"defaultTheme": instance.default_theme,
"darkTheme": ast.literal_eval(instance.dark_theme),
"lightTheme": ast.literal_eval(instance.light_theme),
}
self.save_json(json_content, 'shared/theme-config.json', storage)
def create_category_data_json(self, model_instance, storage):
categories = []
instance_objects = model_instance.objects.all()
if not instance_objects.exists():
self.save_json([], 'category/category-metadata.json', storage)
else:
for eachInstance in instance_objects:
instance_data = {
"id": str(eachInstance.category_id),
"name": eachInstance.name,
"coverImage": self.sanitize_media_link(eachInstance.cover_image),
"tagLine": eachInstance.tagline,
"description": eachInstance.description,
"featuredBlog": eachInstance.featured_id
}
categories.append(instance_data)
self.create_instance_data(instance_data, model_instance.objects.get(category_id=eachInstance.category_id), storage)
self.save_json(categories, 'category/category-metadata.json', storage)
def create_blog_data_json(self, instance, storage):
print(instance)
if not instance.exists():
pass
else:
for eachBlog in instance:
instance_data = {
"id": str(eachBlog.blog_id),
"name": eachBlog.name,
"description": eachBlog.description,
"coverimage": self.sanitize_media_link(eachBlog.cover_image),
"tagLine": eachBlog.tagline,
"parentCategory": str(eachBlog.parent_category.category_id),
"contentBody": eachBlog.content_body
}
self.save_json(instance_data, f'blog/{instance_data["id"]}/blog-data.json', storage)
def merge_media(self):
source_dir = 'media/data'
destination_dir = 'deploy/data'
if not os.path.exists(source_dir):
print("The source directory does not exist.")
else:
try:
shutil.copytree(source_dir, destination_dir, dirs_exist_ok=True)
print(f"Directory copied successfully from {source_dir} to {destination_dir}")
except Exception as e:
print(f"Error occurred: {e}")
def create_instance_data(self, instance_data, blogs_by_category_instance, storage):
instance_data["blogMetadata"]=[]
blogs = blogs_by_category_instance.blogs.all()
if not blogs.exists():
self.save_json(instance_data, f'category/{instance_data["id"]}/category-data.json', storage)
else:
for eachBlog in blogs:
instance_data["blogMetadata"].append({
"id": str(eachBlog.blog_id),
"name": eachBlog.name,
"description": eachBlog.description,
"coverimage": self.sanitize_media_link(eachBlog.cover_image),
"tagLine": eachBlog.tagline,
"parentCategory": instance_data["id"]
})
self.save_json(instance_data, f'category/{instance_data["id"]}/category-data.json', storage)
def save_json(self, json_content, file_name, storage):
data_json = json.dumps(json_content, indent=2)
if storage.exists(file_name):
storage.delete(file_name)
storage.save(file_name, ContentFile(data_json.encode('utf-8')))
def sanitize_media_link(self, string):
if not string:
return ''
return string.replace('http://127.0.0.1:8000/media/data/', '')

1
frontend/.gitignore vendored
View File

@ -9,6 +9,7 @@ lerna-debug.log*
node_modules
dist
public/data
dist-ssr
*.local

View File

@ -166,10 +166,7 @@ function Blog(props) {
<Col xs="3" className="d-none d-md-block"></Col>
<Col className={`blogContent ${ThemeConfig[GlobalTheme].textColor}`} style={{marginBottom: '25px'}}>
<EditorComponent notificationToggler={props.notificationToggler} setContent={setBlogContent} GlobalTheme={GlobalTheme} ThemeConfig={ThemeConfig} content={blogContent} resourceType='blog' resourceId={blogData.id}/>
<ButtonGroup className='mt-4'>
<Button onClick={(event) => setInfo(event)} color={ThemeConfig[GlobalTheme].buttonColor} outline>Save Data</Button>
<Button color={ThemeConfig[GlobalTheme].buttonColor} outline>Publish Data</Button>
</ButtonGroup>
<Button className='mt-3 mb-2' onClick={(event) => setInfo(event)} color={ThemeConfig[GlobalTheme].buttonColor} outline>Save Data</Button>
</Col>
<Col xs="3" className="d-none d-md-block"></Col>
</Row>

View File

@ -140,10 +140,7 @@ function Blogs(props) {
itemObject={item}
/>
)) : <Spinner />}
<ButtonGroup className='mt-4'>
<Button onClick={() => updateInfo()} color={ThemeConfig[GlobalTheme].buttonColor} outline>Save Data</Button>
<Button color={ThemeConfig[GlobalTheme].buttonColor} outline>Publish Data</Button>
</ButtonGroup>
<Button className='mt-3 mb-2' onClick={() => updateInfo()} color={ThemeConfig[GlobalTheme].buttonColor} outline>Save Data</Button>
</div>
</Col>

View File

@ -7,7 +7,6 @@ import EditableMediaService from '../../services/editable-media-service'
function HomePage(props) {
const [introContent, setIntroContent] = useState("")
const [profilePhoto, setProfilePhoto] = useState(false)
const [saveKeyReady, setSaveKeyReady] = useState(true)
const [nameFieldInvalid, setNameFieldInvalid] = useState(false)
const [modal, setModal] = useState(false)
const nameField = useRef(null)
@ -61,7 +60,7 @@ function HomePage(props) {
<Row className='d-md-block'>
<Col xs="4" className="d-none d-md-block"></Col>
{UserData.profilePhoto !== "" && (
<center><img style={{ width: '180px', height: '180px', objectFit: 'cover' }} className="rounded-circle" src={EditableMediaService.getMedia(UserData.profilePhoto)} alt="Profile Photo" /></center>
<center><img style={{ width: '180px', height: '180px', objectFit: 'cover' }} className="mt-5 mb-2 rounded-circle" src={EditableMediaService.getMedia(UserData.profilePhoto)} alt="Profile Photo" /></center>
)}
<Col xs="4" className="d-none d-md-block"></Col>
</Row>
@ -97,10 +96,7 @@ function HomePage(props) {
</FormFeedback>}
</InputGroup>
<EditorComponent notificationToggler={props.notificationToggler} GlobalTheme={GlobalTheme} ThemeConfig={ThemeConfig} content={UserData.introContent} setContent={setIntroContent} resourceType='homepage' resourceId='homepage' />
<ButtonGroup className={`mt-4`}>
<Button onClick={() => setInfo()} color={ThemeConfig[GlobalTheme].buttonColor} outline>Save Data</Button>
<Button color={ThemeConfig[GlobalTheme].buttonColor} outline>Publish Data</Button>
</ButtonGroup>
<Button className='mt-3 mb-2' onClick={() => setInfo()} color={ThemeConfig[GlobalTheme].buttonColor} outline>Save Data</Button>
</Col>
<Col xs="3" className="d-none d-md-block"></Col>

View File

@ -152,6 +152,7 @@ function Header(props) {
outline
onClick={() => setInfo(null, null, 'darkTheme')}>Dark Theme</Button>
</ButtonGroup>
<Button color={`${ThemeConfig ? ThemeConfig[GlobalTheme].navBar['buttonColor'] : ""}`} className='ms-5' outline>Publish Data</Button>
</NavItem>
</Nav>
</Container>

View File

@ -65,7 +65,7 @@ function Blog(props) {
<Row className="mb-4">
<Col className="p-0">
{
blogData.coverImage !== "" ?
blogData.coverImage ?
<img
src={MediaService.getMedia(blogData.coverImage)}
alt="Banner"

View File

@ -19,7 +19,7 @@ function CardListViewer(props) {
if (props.totalItems > 0 && itemObject && Object.keys(itemObject).length !== 0)
return (
<Card color={props.borderColor} outline className={`my-2 ${props.bgColor}`} style={{"width": props.cardType === "smallCard" ? "18rem": "100%"}}>
{itemObject.coverImage !== "" ? <CardImg src={MediaService.getMedia(itemObject.coverImage)} style={{ "height": "180px", "objectFit": "cover" }} top width="100%" /> : ""}
{itemObject.coverImage ? <CardImg src={MediaService.getMedia(itemObject.coverImage)} style={{ "height": "180px", "objectFit": "cover" }} top width="100%" /> : ""}
<CardBody>
<Link to={`/${props.resourceType}/${itemObject.id}`}>
<CardTitle className={`${props.textColor}`} tag="h5">