converted comment section and the cannel info and the action buttons to components

This commit is contained in:
ghassan 2026-03-21 03:22:30 +03:00
parent 84fcbd84dc
commit 64eadfaf56
10 changed files with 1324 additions and 11985 deletions

View File

@ -17,6 +17,7 @@ class CommentController extends Controller
public function index(Video $video)
{
$comments = $video->comments()->whereNull('parent_id')->with(['user', 'replies.user'])->get();
return response()->json($comments);
}
@ -40,7 +41,9 @@ class CommentController extends Controller
// For now, we just parse them
}
return response()->json($comment->load('user'));
$video->increment('comment_count');
return response()->json(['success' => true, 'comment' => $comment->load('user')]);
}
public function update(Request $request, Comment $comment)
@ -67,6 +70,7 @@ class CommentController extends Controller
}
$comment->delete();
return response()->json(['success' => true]);
}
}

View File

@ -1,19 +1,24 @@
<div class="comments-section" style="margin-top: 24px; padding-top: 16px; border-top: 1px solid var(--border-color);">
<h3 style="font-size: 18px; font-weight: 600; margin-bottom: 16px;">
Comments <span style="color: var(--text-secondary); font-weight: 400;">({{ $video->comment_count }})</span>
Comments <span style="color: var(--text-secondary); font-weight: 400;"
id="commentCount{{ $video->id }}">({{ isset($video->comment_count) ? $video->comment_count : 0 }})</span>
</h3>
@auth
<div class="comment-form" style="display: flex; gap: 12px; margin-bottom: 24px;">
<img src="{{ Auth::user()->avatar_url }}" class="channel-avatar" style="width: 40px; height: 40px;"
alt="{{ Auth::user()->name }}">
<div style="flex: 1; display: flex; align-items: center; gap: 8px;">
<textarea id="commentBody{{ $video->id }}" class="form-control"
placeholder="Add a comment... Use @ to mention someone" rows="1"
style="background: transparent; border: none; border-bottom: 2px solid var(--border-color); color: var(--text-primary); border-radius: 0; padding: 12px 0 8px 0; flex: 1; margin: 0; height: 40px; font-size: 14px; outline: none; overflow: hidden;"></textarea>
placeholder="Add a comment... Use @mm.ss for timestamps (e.g. @1.30)" rows="1"
style="background: transparent; border: none; border-bottom: 2px solid var(--border-color); color: var(--text-primary); border-radius: 0; padding: 12px 0 8px 0; flex: 1; margin: 0; height: 40px; font-size: 14px; outline: none; overflow: hidden; resize: none;"></textarea>
<button type="button" class="action-btn" onclick="clearCommentForm('{{ $video->id }}')"
style="flex-shrink: 0;">
<i class="bi bi-x-lg"></i>
</button>
<button type="button" class="action-btn comment-btn" onclick="submitComment({{ $video->id }})"
style="flex-shrink: 0;">
<span>Send</span>
<i class="bi bi-chat-dots"></i>
<span>Comment</span>
</button>
</div>
</div>
@ -23,19 +28,62 @@
<a href="{{ route('login') }}" style="color: var(--brand-red);">Sign in</a> to comment
</div>
@endauth
<div id="commentsList{{ $video->id }}">
@forelse($video->comments()->whereNull('parent_id')->with('user', 'replies.user')->latest()->get() as $comment)
@if (isset($video))
@forelse($video->comments()->whereNull('parent_id')->with('user', 'replies.user')->latest()->limit(20)->get() as $comment)
@include('videos.partials.comment', ['comment' => $comment])
@empty
<p style="color: var(--text-secondary); text-align: center; padding: 20px;">No comments yet. Be the first to
comment!</p>
<p style="color: var(--text-secondary); text-align: center; padding: 20px;">No comments yet. Be the
first to comment!</p>
@endforelse
@endif
</div>
</div>
<style>
/* Comment Section Specific Styles */
.comment-time-badge {
display: inline-flex;
align-items: center;
gap: 4px;
padding: 2px 7px;
border-radius: 999px;
background: rgba(62, 166, 255, 0.15);
border: 1px solid rgba(62, 166, 255, 0.4);
color: #7dd3fc;
font-weight: 600;
font-size: 12px;
line-height: 1.2;
cursor: pointer;
text-decoration: none;
user-select: none;
margin: 0 2px;
transition: all 0.2s ease;
}
.comment-time-badge:hover {
background: rgba(62, 166, 255, 0.26);
border-color: rgba(125, 211, 252, 0.8);
color: #e0f2fe;
transform: translateY(-1px);
}
.comment-form textarea:focus {
border-bottom-color: var(--brand-red);
}
.action-btn.comment-btn:hover {
background: #dc2626 !important;
}
</style>
<script>
function submitComment(videoId) {
(function() {
const videoId = {{ $video->id ?? 0 }};
const scopePrefix = 'comments_' + videoId;
// Scoped functions
window[scopePrefix + '_submitComment'] = function() {
const body = document.getElementById('commentBody' + videoId).value.trim();
if (!body) return;
@ -43,28 +91,33 @@
method: 'POST',
headers: {
'Content-Type': 'application/json',
'X-CSRF-TOKEN': '{{ csrf_token() }}'
'X-CSRF-TOKEN': document.querySelector('meta[name="csrf-token"]').getAttribute(
'content')
},
body: JSON.stringify({
body: body
body
})
})
.then(response => response.json())
.catch(error => {
console.error('Error:', error);
alert('Failed to post comment: ' + error);
})
.then(data => {
if (data & amp; & amp; data.success) {
if (data?.success) {
document.getElementById('commentBody' + videoId).value = '';
addCommentToList(data.comment, videoId);
window[scopePrefix + '_addCommentToList'](data.comment);
} else {
alert('Failed to post comment');
}
})
.catch(error => {
console.error('Error:', error);
alert('Failed to post comment');
});
}
};
function addCommentToList(comment, videoId) {
window[scopePrefix + '_clearCommentForm'] = function(vid) {
document.getElementById('commentBody' + vid).value = '';
};
window[scopePrefix + '_addCommentToList'] = function(comment) {
const commentsList = document.getElementById('commentsList' + videoId);
const commentHtml = `
<div class="comment-item" style="display: flex; gap: 12px; margin-bottom: 16px;" id="comment-${comment.id}">
@ -74,8 +127,8 @@
<span style="font-weight: 600; font-size: 14px;">${comment.user.name}</span>
<span style="color: var(--text-secondary); font-size: 12px;">just now</span>
</div>
<div class="comment-body" style="font-size: 14px; line-height: 1.5; word-wrap: break-word;">
${comment.body.replace(/@(\\\\w+)/g, '<span style="color: #3ea6ff; font-weight: 500;">@$1</span>')}
<div class="comment-body" style="font-size: 14px; line-height: 1.5; word-wrap: break-word;" data-time-enhanced="0">
${escapeHtml(comment.body)}
</div>
<div style="display: flex; gap: 12px; margin-top: 8px;">
<button onclick="toggleReplyForm(${comment.id})" style="background: none; border: none; color: var(--text-secondary); font-size: 12px; font-weight: 600; cursor: pointer; padding: 0;">
@ -86,65 +139,137 @@
</div>
`;
commentsList.insertAdjacentHTML('afterbegin', commentHtml);
const commentCount = commentsList.closest('.comments-section').querySelector('h3 span');
if (commentCount) {
const count = parseInt(commentCount.textContent.match(/\\((\\d+)\\)/)?.[2] || 0) + 1;
commentCount.textContent = `(${count})`;
window[scopePrefix + '_enhanceCommentBodyWithTimeBadges'](commentsList);
window[scopePrefix + '_updateCommentCount'](1);
};
window[scopePrefix + '_updateCommentCount'] = function(delta = 0) {
const countEl = document.getElementById('commentCount' + videoId);
if (countEl) {
let count = parseInt(countEl.textContent.match(/\((\d+)\)/)?.[1] || 0) + delta;
countEl.textContent = `(${Math.max(0, count)})`;
}
};
window[scopePrefix + '_parseDotTimeToSeconds'] = function(dotTime) {
const parts = String(dotTime).trim().split('.');
if (parts.length !== 2) return null;
const mins = parseInt(parts[0], 10);
const secs = parseInt(parts[1], 10);
if (isNaN(mins) || isNaN(secs) || secs < 0 || secs > 59) return null;
return mins * 60 + secs;
};
let commentPlaybackStopHandler = null;
let commentPlaybackEndTime = null;
window[scopePrefix + '_clearCommentPlaybackHandler'] = function(videoPlayer) {
if (videoPlayer && commentPlaybackStopHandler) {
videoPlayer.removeEventListener('timeupdate', commentPlaybackStopHandler);
}
commentPlaybackStopHandler = null;
commentPlaybackEndTime = null;
};
window[scopePrefix + '_playCommentTimeRange'] = function(startSec, endSec = null) {
const videoPlayer = document.querySelector('#videoPlayer, video');
if (!videoPlayer) return;
window[scopePrefix + '_clearCommentPlaybackHandler'](videoPlayer);
const startPlayback = () => {
const playbackStart = Math.max(0, startSec - 1);
videoPlayer.currentTime = playbackStart;
videoPlayer.play().catch(e => console.warn('Autoplay prevented:', e));
if (endSec !== null && endSec > startSec) {
commentPlaybackEndTime = endSec;
commentPlaybackStopHandler = () => {
if (videoPlayer.currentTime >= commentPlaybackEndTime) {
videoPlayer.pause();
window[scopePrefix + '_clearCommentPlaybackHandler'](videoPlayer);
}
};
videoPlayer.addEventListener('timeupdate', commentPlaybackStopHandler);
}
};
startPlayback();
};
window[scopePrefix + '_enhanceCommentBodyWithTimeBadges'] = function(root = document) {
const commentBodies = root.querySelectorAll('.comment-body[data-time-enhanced="0"]');
const timeRangeRegex = /@(\\d{1,2}\\.\\d{2})(?:-(\\d{1,2}\\.\\d{2}))?/g;
const mentionRegex = /@(\\w+)/g;
commentBodies.forEach(bodyEl => {
const originalText = bodyEl.textContent || '';
if (!originalText.trim()) return;
let html = originalText.replace(timeRangeRegex, (match, start, end) => {
const startSec = window[scopePrefix + '_parseDotTimeToSeconds'](start);
const endSec = end ? window[scopePrefix + '_parseDotTimeToSeconds'](end) :
null;
if (startSec === null || (end && endSec === null)) return match;
if (endSec !== null && endSec <= startSec) return match;
const label = end ? `@${start}-${end}` : `@${start}`;
return `<span class="comment-time-badge" data-start="${startSec}" data-end="${endSec ?? ''}">${label}</span>`;
});
html = html.replace(mentionRegex, (m, u) => `@${u}`);
bodyEl.innerHTML = html.replace(/(^|[\\s>])@(\\w+)/g,
'$1<span style="color: #3ea6ff; font-weight: 500;">@$2</span>');
bodyEl.dataset.timeEnhanced = '1';
});
root.querySelectorAll('.comment-time-badge').forEach(badge => {
if (badge.dataset.bound === '1') return;
badge.addEventListener('click', function(e) {
e.preventDefault();
e.stopPropagation();
const start = parseFloat(this.dataset.start || '');
const end = this.dataset.end === '' ? null : parseFloat(this.dataset.end);
if (!isNaN(start)) {
window[scopePrefix + '_playCommentTimeRange'](start, end);
}
});
badge.dataset.bound = '1';
});
};
function escapeHtml(text) {
const div = document.createElement('div');
div.textContent = text;
return div.innerHTML;
}
function deleteComment(commentId) {
if (confirm('Are you sure you want to delete this comment?')) {
// Global functions for partials (edit/delete/reply)
window.submitComment = window[scopePrefix + '_submitComment'];
window.clearCommentForm = window[scopePrefix + '_clearCommentForm'];
window.deleteComment = function(commentId) {
if (confirm('Are you sure?')) {
fetch(`/comments/${commentId}`, {
method: 'DELETE',
headers: {
'X-CSRF-TOKEN': '{{ csrf_token() }}'
'X-CSRF-TOKEN': document.querySelector('meta[name="csrf-token"]').getAttribute(
'content')
}
})
.then(response => response.json())
.then(res => res.json())
.then(data => {
if (data.success) {
document.getElementById('comment-' + commentId).remove();
// Update comment count for all comment sections (simplified)
document.querySelectorAll('.comments-section h3 span').forEach(span => {
const count = parseInt(span.textContent.match(/\\((\\d+)\\)/)?.[2] || 0) - 1;
span.textContent = `(${Math.max(0, count)})`;
});
document.getElementById('comment-' + commentId)?.remove();
window[scopePrefix + '_updateCommentCount'](-1);
}
});
}
}
};
document.addEventListener('DOMContentLoaded', function() {
const commentTexts = document.querySelectorAll('.comment-body');
commentTexts.forEach(text => {
const html = text.innerHTML.replace(/@(\\w+)/g,
'<span style="color: #3ea6ff; font-weight: 500;">@$1</span>');
text.innerHTML = html;
// Initialize
document.addEventListener('DOMContentLoaded', () => {
window[scopePrefix + '_enhanceCommentBodyWithTimeBadges']();
});
});
function submitReply(videoId, parentId) {
const textarea = document.querySelector(`#replyForm${parentId} textarea`);
const body = textarea.value.trim();
if (!body) return;
fetch(`/videos/${videoId}/comments`, {
method: 'POST',
headers: {
'Content-Type': 'application/json',
'X-CSRF-TOKEN': '{{ csrf_token() }}'
},
body: JSON.stringify({
body: body,
parent_id: parentId
})
})
.then(response => response.json())
.then(data => {
if (data.success) {
location.reload();
}
});
}
})();
</script>

View File

@ -601,108 +601,9 @@
</style>
@endif
<!-- Comments Section -->
<div class="comments-section"
style="margin-top: 24px; padding-top: 16: 1px solid var(--borderpx; border-top-color);">
<h3 style="font-size: 18px; font-weight: 600; margin-bottom: 16px;">
Comments <span
style="color: var(--text-secondary); font-weight: 400;">({{ $video->comment_count }})</span>
</h3>
@auth
<div class="comment-form" style="display: flex; gap: 12px; margin-bottom: 24px;">
<img src="{{ Auth::user()->avatar_url }}" class="channel-avatar" style="width: 40px; height: 40px;"
alt="{{ Auth::user()->name }}">
<div style="flex: 1;">
<textarea id="commentBody" class="form-control" placeholder="Add a comment... Use @ to mention someone"
rows="3"
style="background: var(--bg-secondary); border: 1px solid var(--border-color); color: var(--text-primary); border-radius: 8px; padding: 12px; width: 100%; resize: none;"></textarea>
<div style="display: flex; gap: 8px; margin-top: 8px; justify-content: flex-end;">
<button type="button" class="yt-action-btn"
onclick="document.getElementById('commentBody').value = ''">Cancel</button>
<button type="button" class="yt-action-btn"
style="background: var(--brand-red); color: white;"
onclick="submitComment({{ $video->id }})">Comment</button>
@include('components.video-comments', ['video' => $video])
</div>
</div>
</div>
@else
<div
style="margin-bottom: 24px; padding: 16px; background: var(--bg-secondary); border-radius: 8px; text-align: center;">
<a href="{{ route('login') }}" style="color: var(--brand-red);">Sign in</a> to comment
</div>
@endauth
<div id="commentsList">
@forelse($video->comments()->whereNull('parent_id')->with('user', 'replies.user')->latest()->get() as $comment)
@include('videos.partials.comment', ['comment' => $comment])
@empty
<p style="color: var(--text-secondary); text-align: center; padding: 20px;">No comments yet. Be the
first to comment!</p>
@endforelse
</div>
</div>
<script>
function submitComment(videoId) {
const body = document.getElementById('commentBody').value.trim();
if (!body) return;
fetch(`/videos/${videoId}/comments`, {
method: 'POST',
headers: {
'Content-Type': 'application/json',
'X-CSRF-TOKEN': '{{ csrf_token() }}'
},
body: JSON.stringify({
body: body
})
}).then(r => r.json()).then(data => {
if (data.success) {
document.getElementById('commentBody').value = '';
location.reload();
}
});
}
function deleteComment(commentId) {
if (!confirm('Delete this comment?')) return;
fetch(`/comments/${commentId}`, {
method: 'DELETE',
headers: {
'X-CSRF-TOKEN': '{{ csrf_token() }}'
}
})
.then(r => r.json()).then(data => {
if (data.success) location.reload();
});
}
document.addEventListener('DOMContentLoaded', function() {
document.querySelectorAll('.comment-body').forEach(text => {
text.innerHTML = text.innerHTML.replace(/@(\w+)/g,
'<span style="color: #3ea6ff; font-weight: 500;">@$1</span>');
});
});
function submitReply(videoId, parentId) {
const textarea = document.querySelector(`#replyForm${parentId} textarea`);
const body = textarea.value.trim();
if (!body) return;
fetch(`/videos/${videoId}/comments`, {
method: 'POST',
headers: {
'Content-Type': 'application/json',
'X-CSRF-TOKEN': '{{ csrf_token() }}'
},
body: JSON.stringify({
body: body,
parent_id: parentId
})
}).then(r => r.json()).then(data => {
if (data.success) location.reload();
});
}
</script>
</div>
<!-- Sidebar - Up Next / Recommendations -->
<div class="yt-sidebar-container">
@ -728,8 +629,7 @@
alt="{{ $playlistVideo->title }}">
@endif
@if ($playlistVideo->duration)
<span
class="yt-video-duration">{{ gmdate('i:s', $playlistVideo->duration) }}</span>
<span class="yt-video-duration">{{ gmdate('i:s', $playlistVideo->duration) }}</span>
@endif
@if ($playlistVideo->is_shorts)
<span class="yt-shorts-badge"

File diff suppressed because it is too large Load Diff

View File

@ -1,9 +0,0 @@
@extends('layouts.app')
@section('title', ($video->title ?? 'Match Video') . ' | ' . config('app.name'))
@section('extra_styles')
<style>
/* ===== Video Section ===== */
.yt-video-section {
flex: 1;
min

View File

@ -2326,9 +2326,7 @@
</script>
@endif
<!-- Comment Section -->
<div class="comments-section"
style="margin-top: 24px; padding-top: 16px; border-top: 1px solid var(--border-color);">
<x-video-comments :video="$video" />
<h3 style="font-size: 18px; font-weight: 600; margin-bottom: 16px;">
Comments <span
style="color: var(--text-secondary); font-weight: 400;">({{ isset($video) ? $video->comment_count ?? 0 : 0 }})</span>
@ -2341,8 +2339,8 @@
<textarea id="commentBody" class="form-control" placeholder="Add a comment... Use @ to mention someone"
rows="1"
style="background: transparent; border: none; border-bottom: 2px solid var(--border-color); color: var(--text-primary); border-radius: 0; padding: 12px 0 8px 0; flex: 1; margin: 0; height: 40px; font-size: 14px; outline: none; overflow: hidden;"></textarea>
<button type="button" class="action-btn"
onclick="document.getElementById('commentBody').value = ''" style="flex-shrink: 0;">
<button type="button" class="action-btn" onclick="document.getElementById('commentBody').value = ''"
style="flex-shrink: 0;">
<i class="bi bi-x-lg"></i>
<span>Cancel</span>
</button>
@ -2602,8 +2600,7 @@
<div class="tab-panels">
<!-- Points Tab -->
<div class="tab-panel active" id="tab-official">
<div
style="display: flex; justify-content: space-between; align-items: center; margin-bottom: 4px;">
<div style="display: flex; justify-content: space-between; align-items: center; margin-bottom: 4px;">
<div>
<div class="section-label">Rounds & points</div>
</div>
@ -2777,8 +2774,7 @@
<!-- Coach Review Tab -->
<div class="tab-panel" id="tab-review">
<div
style="display: flex; justify-content: space-between; align-items: center; margin-bottom: 4px;">
<div style="display: flex; justify-content: space-between; align-items: center; margin-bottom: 4px;">
<div class="section-label">Private notes</div>
@auth
@if (isset($video) && Auth::id() === $video->user_id)
@ -2820,8 +2816,8 @@
<div class="review-content">
<h4 class="review-note-title">Missed counter opportunity</h4>
<div class="review-author-bar">
<img class="review-author-avatar"
src="https://picsum.photos/seed/coach-sara/80/80" alt="Coach Sara">
<img class="review-author-avatar" src="https://picsum.photos/seed/coach-sara/80/80"
alt="Coach Sara">
<span class="review-author-name">Coach Sara</span>
<span></span>
<span class="review-note-text">Great angle, but no follow up.</span>
@ -2846,8 +2842,8 @@
<div class="review-content">
<h4 class="review-note-title">Excellent angle change and follow-up</h4>
<div class="review-author-bar">
<img class="review-author-avatar"
src="https://picsum.photos/seed/coach-ahmed/80/80" alt="Coach Ahmed">
<img class="review-author-avatar" src="https://picsum.photos/seed/coach-ahmed/80/80"
alt="Coach Ahmed">
<span class="review-author-name">Coach Ahmed</span>
<span></span>
<span class="review-note-text">Ideal example of exit and re-entry.</span>
@ -2946,8 +2942,7 @@
alt="{{ $recVideo->title }}">
@endif
@if ($recVideo->duration)
<span
class="yt-video-duration">{{ gmdate('i:s', $recVideo->duration) }}</span>
<span class="yt-video-duration">{{ gmdate('i:s', $recVideo->duration) }}</span>
@endif
@if ($recVideo->is_shorts)
<span class="yt-shorts-badge"

View File

@ -210,13 +210,81 @@
color: var(--text-secondary);
}
.video-actions {
display: flex;
align-items: center;
gap: 8px;
}
.yt-action-btn {
display: flex;
align-items: center;
gap: 8px;
padding: 8px 16px;
border-radius: 20px;
border: none;
background: var(--bg-secondary);
color: var(--text-primary);
cursor: pointer;
font-size: 14px;
font-weight: 500;
}
.yt-action-btn:hover {
background: var(--border-color);
}
.action-btn,
.comments-section .action-btn {
border: none;
border-radius: 8px;
padding: 8px 14px;
font-size: 0.82rem;
font-weight: 500;
cursor: pointer;
background: var(--bg-secondary);
border: 1px solid var(--border-color);
color: var(--text-primary);
display: flex;
align-items: center;
gap: 6px;
transition: all 0.2s ease;
}
.action-btn:hover,
.comments-section .action-btn:hover {
background: var(--border-color);
transform: translateY(-1px);
}
.action-btn:active,
.comments-section .action-btn:active {
transform: translateY(0);
}
.action-btn svg,
.action-btn i,
.comments-section .action-btn svg,
.comments-section .action-btn i {
flex-shrink: 0;
}
.action-btn.comment-btn {
background: var(--brand-red);
color: white;
border-color: var(--brand-red);
}
.yt-action-btn.liked {
color: var(--brand-red);
}
/* Channel Row */
.channel-row {
display: flex;
align-items: center;
justify-content: space-between;
padding: 16px 0;
overflow: visible;
}
.channel-info {
@ -242,6 +310,18 @@
color: var(--text-secondary);
}
.subscribe-btn {
background: white;
color: black;
border: none;
padding: 8px 16px;
border-radius: 18px;
font-weight: 600;
font-size: 14px;
cursor: pointer;
white-space: nowrap;
}
/* Description */
.video-description {
background: var(--bg-secondary);
@ -352,16 +432,8 @@
.video-actions {
width: 100%;
justify-content: flex-end;
}
.video-actions>.desktop-action {
display: none !important;
}
.mobile-action-dropdown {
display: block;
margin-left: auto;
overflow-x: auto;
justify-content: flex-start;
}
.yt-main {
@ -474,7 +546,76 @@
</div>
</div>
<x-channel-row :video="$video" />
<!-- Channel Row - All in one line -->
<div class="channel-row"
style="display: flex; align-items: center; justify-content: space-between; padding: 12px 0; flex-wrap: wrap; gap: 16px;">
<div style="display: flex; align-items: center; gap: 12px;">
<a href="{{ route('channel', $video->user_id) }}" class="channel-info text-decoration-none"
style="color: inherit; display: flex; align-items: center; gap: 12px;">
@if ($video->user)
<img src="{{ $video->user->avatar_url }}" class="channel-avatar"
style="width: 40px; height: 40px; border-radius: 50%;" alt="{{ $video->user->name }}">
@else
<div class="channel-avatar" style="width: 40px; height: 40px; border-radius: 50%;"></div>
@endif
<div>
<div class="channel-name" style="font-weight: 600;">{{ $video->user->name ?? 'Unknown' }}</div>
<div class="channel-subs" style="font-size: 12px; color: var(--text-secondary);">
{{ number_format($video->user->subscriber_count ?? 0) }} subscribers</div>
</div>
</a>
</div>
<div class="video-actions" style="display: flex; align-items: center; gap: 8px; flex-wrap: wrap;">
@auth
@if (Auth::id() !== $video->user_id)
<button class="subscribe-btn">Subscribe</button>
@else
<button class="action-btn" onclick="openEditVideoModal({{ $video->id }})">
<i class="bi bi-pencil"></i>
<span>Edit</span>
</button>
@endif
@else
<a href="{{ route('login') }}" class="subscribe-btn">Subscribe</a>
@endauth
@auth
<form method="POST"
action="{{ $video->isLikedBy(Auth::user()) ? route('videos.unlike', $video->id) : route('videos.like', $video->id) }}"
class="d-inline">
@csrf
<button type="submit" class="action-btn {{ $video->isLikedBy(Auth::user()) ? 'liked' : '' }}">
<i
class="bi {{ $video->isLikedBy(Auth::user()) ? 'bi-hand-thumbs-up-fill' : 'bi-hand-thumbs-up' }}"></i>
{{ $video->like_count > 0 ? number_format($video->like_count) : 'Like' }}
</button>
</form>
@else
<a href="{{ route('login') }}" class="yt-action-btn">
<i class="bi bi-hand-thumbs-up"></i> Like
</a>
@endauth
@if ($video->isShareable())
<button class="action-btn"
onclick="openShareModal('{{ $video->share_url }}', '{{ addslashes($video->title) }}')">
<i class="bi bi-share"></i> Share
</button>
@endif
<!-- Save to Playlist Button -->
<button class="action-btn" onclick="openAddToPlaylistModal({{ $video->id }})">
<svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor"
stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
<path d="M19 21l-7-5-7 5V5a2 2 0 0 1 2-2h10a2 2 0 0 1 2 2z"></path>
</svg>
<span>Save</span>
</button>
</div>
</div>
<!-- Description Box -->
@if ($video->description)
@ -546,10 +687,49 @@
</style>
@endif
<x-video-comments :video="$video" />
<!-- Comment Section -->
<div class="comments-section"
style="margin-top: 24px; padding-top: 16px; border-top: 1px solid var(--border-color);">
<h3 style="font-size: 18px; font-weight: 600; margin-bottom: 16px;">
Comments <span
style="color: var(--text-secondary); font-weight: 400;">({{ $video->comment_count }})</span>
</h3>
@auth
<div class="comment-form" style="display: flex; gap: 12px; margin-bottom: 24px;">
<img src="{{ Auth::user()->avatar_url }}" class="channel-avatar" style="width: 40px; height: 40px;"
alt="{{ Auth::user()->name }}">
<div style="flex: 1; display: flex; align-items: center; gap: 8px;">
<textarea id="commentBody" class="form-control" placeholder="Add a comment... Use @ to mention someone"
rows="1"
style="background: transparent; border: none; border-bottom: 2px solid var(--border-color); color: var(--text-primary); border-radius: 0; padding: 12px 0 8px 0; flex: 1; margin: 0; height: 40px; font-size: 14px; outline: none; overflow: hidden;"></textarea>
<button type="button" class="action-btn"
onclick="document.getElementById('commentBody').value = ''" style="flex-shrink: 0;">
<i class="bi bi-x-lg"></i>
<span>Cancel</span>
</button>
<button type="button" class="action-btn comment-btn"
onclick="submitComment({{ $video->id }})" style="flex-shrink: 0;">
<i class="bi bi-chat-dots"></i>
<span>Comment</span>
</button>
</div>
</div>
@else
<div
style="margin-bottom: 24px; padding: 16px; background: var(--bg-secondary); border-radius: 8px; text-align: center;">
<a href="{{ route('login') }}" style="color: var(--brand-red);">Sign in</a> to comment
</div>
@endauth
<div id="commentsList">
@forelse($video->comments()->whereNull('parent_id')->with('user', 'replies.user')->latest()->get() as $comment)
@include('videos.partials.comment', ['comment' => $comment])
@empty
<p style="color: var(--text-secondary); text-align: center; padding: 20px;">No comments yet. Be the
first to comment!</p>
@endforelse
</div>
</div>
<script>
@ -692,7 +872,8 @@
alt="{{ $playlistVideo->title }}">
@endif
@if ($playlistVideo->duration)
<span class="yt-video-duration">{{ gmdate('i:s', $playlistVideo->duration) }}</span>
<span
class="yt-video-duration">{{ gmdate('i:s', $playlistVideo->duration) }}</span>
@endif
@if ($playlistVideo->is_shorts)
<span class="yt-shorts-badge"

View File

@ -1,45 +0,0 @@
<script>
function showDeleteCommentModal(commentId, commentBody) {
const modalId = 'confirmDeleteModal' + commentId;
const modalElement = document.getElementById(modalId);
if (modalElement) {
const modal = new bootstrap.Modal(modalElement);
modal.show();
window.currentDeleteCommentId = commentId;
}
}
function deleteCommentWithModal(commentId) {
fetch(`/comments/${commentId}`, {
method: 'DELETE',
headers: {
'X-CSRF-TOKEN': document.querySelector('meta[name="csrf-token"]').getAttribute('content'),
'Accept': 'application/json'
}
})
.then(response => response.json())
.then(data => {
if (data.success) {
document.getElementById('comment-' + commentId).remove();
const commentCount = document.querySelector('h3 span');
if (commentCount) {
const count = parseInt(commentCount.textContent.match(/\\((\\d+)\\)/)?.[2] || 0) - 1;
commentCount.textContent = `(${Math.max(0, count)})`;
}
// Close modal
const modalId = 'confirmDeleteModal' + commentId;
const modalElement = document.getElementById(modalId);
if (modalElement) {
const modal = bootstrap.Modal.getInstance(modalElement);
modal.hide();
}
} else {
alert('Failed to delete comment');
}
})
.catch(error => {
console.error('Error:', error);
alert('Failed to delete comment');
});
}
</script>