Hệ thống pháp luật

ỦY BAN NHÂN DÂN
TỈNH ĐỒNG THÁP
-------

CỘNG HÒA XÃ HỘI CHỦ NGHĨA VIỆT NAM
Độc lập - Tự do - Hạnh phúc
---------------

Số: 321/KH-UBND

Đồng Tháp, ngày 04 tháng 10 năm 2023

 

KẾ HOẠCH

TRIỂN KHAI CHƯƠNG TRÌNH PHỐI HỢP NÂNG CAO CHẤT LƯỢNG CHUỖI CUNG ỨNG NÔNG LÂM THỦY SẢN BỀN VỮNG GIAO THƯƠNG GIỮA THÀNH PHỐ HỒ CHÍ MINH VÀ TỈNH ĐỒNG THÁP NĂM 2023

Căn cứ Chương trình phối hợp số 01/CTPH-BNNPTNT-UBND ngày 23/02/2023 giữa Bộ Nông nghiệp và Phát triển nông thôn và Ủy ban nhân dân Thành phố Hồ Chí Minh về nâng cao chất lượng chuỗi cung ứng nông lâm thủy sản bền vững giao thương giữa Thành phố Hồ Chí Minh và các tỉnh, thành phố giai đoạn 2022 - 2025; Công văn số 326/UBND-KT ngày 28/3/2023 của Ủy ban nhân dân Tỉnh về việc triển khai thực hiện Chương trình phối hợp nâng cao chất lượng chuỗi cung ứng nông lâm thủy sản bền vững giao thương giữa Thành phố Hồ Chí Minh và tỉnh Đồng Tháp giai đoạn 2022 - 2025;

Ủy ban nhân dân dân tỉnh Đồng Tháp xây dựng Kế hoạch triển khai Chương trình phối hợp nâng cao chất lượng chuỗi cung ứng nông lâm thủy sản bền vững giao thương giữa Thành phố Hồ Chí Minh và tỉnh Đồng Tháp năm 2023, nội dung cụ thể như sau:

I. MỤC TIÊU

- Từng bước nâng thị phần, sản lượng nông sản, thực phẩm Đồng Tháp tiêu thụ trên địa bàn Thành phố Hồ Chí Minh; xây dựng, tạo mối liên kết giữa sản xuất và tiêu thụ nông sản, thực phẩm an toàn được ổn định, bền vững, đáp ứng nhu cầu tiêu dùng+ Thực hiện tập huấn 06 lớp kiến thức an toàn thực phẩm cho các hợp tác xã, tổ hợp tác, cơ sở sản xuất, kinh doanh thực phẩm (cơ sở) trên địa bàn Tỉnh.

+ Tổ chức 01 lớp bồi dưỡng về nông nghiệp hữu cơ cho người làm công tác tập huấn (ToT) cho cán bộ chuyên môn trên địa bàn Tỉnh.

+ Xây dựng, phát triển, xác nhận 02 chuỗi cung ứng nông sản thực phẩm an toàn (theo Quyết định số 3075/QĐ-BNN-QLCL+ Thực hiện kiểm tra, hậu kiểm 60 cơ sở hoạt động sản xuất, chế biến, kinh doanh, thực phẩm.

+ Thu 20 mẫu nông sản tại các vùng sản xuất cây ăn trái, rau màu tập trung, cơ sở chế biến mẫu thực phẩm nông sản giám sát các chỉ tiêu về an toàn thực phẩm (theo Thông tư số 08/2016/TT-BNNPTNT- Tổ chức được 02 lớp đào tạo tư vấn viên của VietGAP cho cán bộ chuyên môn trên địa bàn Tỉnh. Phấn đấu diện tích áp dụng quy trình thực hành sản xuất nông nghiệp tốt (GAP) đạt 21.822 ha (lúa 16.692 ha, cây ăn trái 4.678 ha, rau màu 452 ha); diện tích thực hiện chứng nhận đủ điều kiện ATTP hoặc ký cam kết sản xuất an toàn đạt 83.343 ha (lúa 66.385 ha, cây ăn trái 14.399 ha, rau màu 2.559 ha).II. CÁC NHIỆM VỤ TRỌNG TÂM

1. Xây dựng, phát triển các chuỗi cung ứng nông sản thực phẩm an toàn giao thương giữa Thành phố Hồ Chí Minh và tỉnh Đồng Tháp

- Xây dựng, phát triển, xác nhận chuỗi cung ứng nông sản thực phẩm an toàn theo Quyết định số 3075/QĐ-BNN-QLCL ngày 20/7/2016 của Bộ Trưởng Bộ Nông nghiệp và Phát triển nông thôn về việc ban hành Hướng dẫn xác nhận sản phẩm chuỗi cung ứng thực phẩm an toàn; phục vụ giao thương thương giữa Thành phố Hồ Chí Minh và tỉnh Đồng Tháp.

- Tăng cường thẩm định, chứng nhận đầy đủ cơ sở đủ điều kiện đảm bảo an toàn thực phẩm theo Thông tư số 38/2018/TT-BNNPTNT ngày 25/12/2018 và Thông tư số 32/2022/TT-BNNPTNT ngày 30/12/2022 của Bộ Nông nghiệp và Phát triển nông thôn sửa đổi, bổ sung một số Thông tư quy định thẩm định, chứng nhận cơ sở sản xuất, kinh doanh thực phẩm nông, lâm, thủy sản đủ điều kiện bảo đảm an toàn thực phẩm thuộc phạm vi quản lý của Bộ nông nghiệp và Phát triển nông thôn.

- Tăng cường vận động cơ sở sản xuất kinh doanh nhỏ lẻ ký cam kết và kiểm tra ký cam kết theo theo Thông tư số 17/2018/TT-BNNPTNT ngày 21/10/2018 của Bộ Nông nghiệp và Phát triển nông thôn.

- Thực hiện tập huấn kiến thức an toàn thực phẩm cho các hợp tác xã, tổ hợp tác, cơ sở sản xuất, kinh doanh thực phẩm (cơ sở) trên địa bàn tỉnh.

- Tổ chức tập huấn, bồi dưỡng cán bộ quản lý chất lượng an toàn thực phẩm ToT về tư vấn, đánh giá nội bộ tiêu chuẩn VietGAP, tiêu chuẩn hữu cơ (xây dựng bộ tài liệu đào tạo, tổ chức các lớp đào tạo, tập huấn).

- Thực hiện công tác giới thiệu và hỗ trợ các cơ sở sản xuất, kinh doanh nông sản tham gia Đề án xây dựng mô hình thí điểm quản lý thực phẩm theo “Chuỗi thực phẩm an toàn” theo Quy chế cấp và thu hồi giấy chứng nhận do UBND Thành phố Hồ Chí Minh ban hành cho các cơ sở sản xuất của tỉnh Đồng Tháp tham gia; phối hợp Ban Quản lý Đề án “Chuỗi thực phẩm an toàn” thẩm định thực tế các cơ sở sản xuất, chăn nuôi, nuôi trồng đã được chứng nhận tham gia Đề án “Chuỗi thực phẩm an toàn” tại tỉnh Đồng Tháp tiêu thụ sản phẩm trên địa bàn Thành phố Hồ Chí Minh.

- Triển khai hướng dẫn cơ sở sản xuất, kinh doanh thực phẩm tươi sống báo cáo định kỳ về số lượng, chủng loại được tiêu thụ trên địa bàn Thành phố Hồ Chí Minh và công tác tự kiểm tra, giám sát quy trình sản xuất, chất lượng của cơ sở.

2. Hỗ trợ, tạo điều kiện thuận lợi cho các cơ sở sản xuất - kinh doanh nông sản, thực phẩm an toàn của tỉnh Đồng Tháp tiêu thụ tại Thành phố Hồ Chí Minh

Nhằm tạo điều kiện quảng bá, giới thiệu sản phẩm, kết nối tiêu thụ, xúc tiến và mở rộng thị trường tiêu thụ sản phẩm nông sản, định kỳ hàng tháng, cung cấp thông tin đăng công khai trên trang thông tin của Sở Nông nghiệp và Phát triển nông thôn:

- Rà soát, thông tin đến các tổ chức, cá nhân, vùng sản xuất có nhu cầu áp dụng tiêu chuẩn VietGAP và hướng dẫn hồ sơ trình tự thủ tục để được hỗ trợ chính sách theo Nghị quyết số 44/2021/NQ-HĐND ngày 17/8/2021 của Hội đồng nhân dân Tỉnh và cấp mã số vùng trồng.

- Thông tin về sản phẩm nông sản đạt chứng nhận GAP, an toàn thực phẩm (ATTP), được cấp mã số vùng trồng (MSVT), tự công bố sản phẩm.

- Cung cấp thông tin, danh sách các chuỗi cung ứng nông sản thực phẩm an toàn có khả năng tiêu thụ sản phẩm tại Thành phố Hồ Chí Minh.

- Công khai, cung cấp thông tin vùng trồng và cơ sở đóng gói đủ điều kiện sản xuất an toàn, cấp mã số vùng trồng và mã số cơ sở đóng gói phục vụ liên kết tiêu thụ sản phẩm.

- Công khai các cơ sở sản xuất trồng trọt ban đầu, các cơ sở chế biến nông sản đủ điều kiện an toàn thực phẩm để các cơ sở có nhu cầu nắm bắt thông tin, tạo liên kết sản xuất, tiêu thụ bền vững.

- Sở Nông nghiệp và Phát triển nông thôn tỉnh Đồng Tháp thường xuyên trao đổi thông tin, phối hợp chặt chẽ với Sở Nông nghiệp và Phát triển nông thôn Thành phố Hồ Chí Minh, Ban Quản lý an toàn thực phẩm Thành phố Hồ Chí Minh trong công tác quản lý chất lượng, an toàn thực phẩm, giám sát, thanh kiểm tra, tuy xuất nguồn gốc sản phẩm nông lâm thủy sản của tỉnh Đồng Tháp tiêu thụ tại Thành phố Hồ Chí Minh và ngược lại.

3. Kiểm tra, giám sát xử lý vi phạm về an toàn thực phẩm

- Thực hiện kiểm tra, hậu kiểm về an toàn thực phẩm thuộc phạm vi quản lý theo quy định pháp luật đối với các hoạt động liên quan đến sản xuất, chế biến, kinh doanh, thực phẩm của cơ sở sản xuất kinh doanh thực phẩm.

- Thực hiện thu mẫu thực phẩm nông sản giám sát các chỉ tiêu về an toàn thực phẩm (theo Thông tư số 08/2016/TT-BNNPTNT) tại các vùng sản xuất cây ăn trái, rau màu tập trung, cơ sở chế biến gửi Phòng kiểm nghiệm phân tích các chỉ tiêu quy định về an toan thực phẩm.

- Chủ động giám sát, cảnh báo, thanh tra, kiểm tra theo Kế hoạch (hoặc đột xuất) nhằm phát hiện kịp thời, xử lý nghiệm cơ sở vi phạm, sản phẩm vi phạm quy định về chất lượng an toàn thực phẩm.

- Định kỳ cung cấp thông tin kết quả giám sát an toàn thực phẩm tại tỉnh Đồng Tháp tiêu thụ trên địa bàn Thành phố Hồ Chí Minh và thông báo các sản phẩm không đảm bảo an toàn thực phẩm để phối hợp điều tra, truy xuất nguồn gốc, xác định nguyên nhân, tăng cường công tác giám sát và xử lý theo quy định.

III. TỔ CHỨC THỰC HIỆN

1. Sở Nông nghiệp và Phát triển nông thôn

- Chủ trì, triển khai thực hiện kế hoạch này; theo dõi, đôn đốc việc triển khai thực hiện.

- Chủ trì, phối hợp tổ chức thanh tra, kiểm tra theo kế hoạch và thanh tra đột xuất cơ sở sản xuất, kinh doanh vật tư nông nghiệp, sản phẩm nông, lâm, thủy sản; tăng cường công tác giám sát, hậu kiểm chất lượng vật tư nông nghiệp, sản phẩm nông lâm thủy sản nhằm kịp thời phát hiện và xử lý nghiêm những trường hợp vi phạm.

- Chỉ đạo các đơn vị trực thuộc thực hiện tốt nội dung Kế hoạch đảm bảo an toàn thực phẩm, nâng cao chất lượng nông lâm thủy sản trong lĩnh vực nông nghiệp năm 2023 theo phân công, phân cấp quản lý.

- Đôn đốc, theo dõi, tổng hợp báo cáo kết quả thực hiện định kỳ hoặc đột xuất gửi Bộ Nông nghiệp và Phát triển nông thôn và Ủy ban nhân dân Tỉnh theo quy định.

- Cung cấp danh sách các chuỗi thực phẩm an toàn, các sản phẩm đã được chứng nhận OCOP và tiềm năng OCOP, sản phẩm chủ lực, các doanh nghiệp, hợp tác xã, cơ sở SXKD nông lâm sản và thủy sản đủ điều kiện trên địa bàn tỉnh Đồng Tháp gửi Sở Nông nghiệp và Phát triển nông thôn Thành phố Hồ Chí Minh để hỗ trợ, kết nối, tiêu thụ sản phẩm.

2. Sở Y tế, Sở Công thương

Phối hợp với Sở Nông nghiệp và Phát triển nông thôn tăng cường thanh tra, quản lý, kiểm soát ngăn chặn việc sản phẩm nông lâm thủy sản chưa đảm bảo an toàn thực phẩm lưu thông trên thị trường.

3. Cục Quản lý thị trường Tỉnh

Tăng cường triển khai đồng bộ các biện pháp nghiệp vụ kiểm soát thị trường; kịp thời bắt giữ, xử lý nghiêm các tổ chức, cá nhân vi phạm, tập trung đấu tranh với hoạt động buôn lậu, vận chuyển trái phép hàng thực phẩm nông lâm thủy sản kém chất lượng, không rõ nguồn gốc xuất xứ, chưa bảo đảm an toàn thực phẩm lưu thông trên thị trường theo đúng quy định của pháp luật.

4. Trung tâm Xúc tiến Thương mại, Du lịch và Đầu tư

Hỗ trợ doanh nghiệp trong Tỉnh tham dự các hội chợ, triển lãm, các diễn đàn thương mại, du lịch, đầu tư nhằm quảng bá, giới thiệu sản phẩm của Tỉnh đến doanh nghiệp, người tiêu dùng trong và ngoài nước.

5. Ủy ban nhân dân các huyện, thành phố

Chỉ đạo các đơn vị chuyên môn tăng cường công tác quản lý nhà nước về an toàn thực phẩm; chủ động phối hợp với các ngành chức năng thành lập đoàn kiểm tra đối với các cơ sở sản xuất kinh doanh sản phẩn nông lâm thủy sản thuộc phạm vi quản lý.

Chỉ đạo các tổ chức đoàn thể, ngành chuyên môn phối hợp triển khai tuyền đến người sản xuất, hợp tác xã, tổ hợp tác, hội quán và nông dân quan tâm thực hiện, tuân thủ các quy trình kỹ thuật đã được phổ biến vào sản xuất của mình, thực hiện ghi chép nhật ký hoặc ứng dụng phần mềm nhật ký điện tử để theo dõi nhằm minh bạch hóa thông tin sản xuất, là cơ sở truy xuất nguồn gốc nguyên liệu cung ứng cho thị trường.

6. Chế độ thông tin báo cáo

Đề nghị các cơ quan có liên quan báo cáo kết quả cả năm thực hiện về Sở Nông nghiệp và Phát triển nông thôn trước ngày 15/12/2023 để tổng hợp báo cáo Ủy ban nhân dân Tỉnh, Bộ Nông nghiệp và Phát triển nông thôn theo quy định (qua Chi cục Trồng trọt và Bảo vệ thực vật, điện thoại: 02773.856969; hộp thư điện tử: phong.qlcl.attp@gmail.com)./.

 


Nơi nhận:
- Bộ NN&PTNT;
- CT và các PCT/UBND Tỉnh;
- UBND TP. HCM;
- Ban Quản lý ATTP TP. HCM;
- Các Sở: NN&PTNT, YT, CT, TC;
- TT. XTTMDL&ĐT;
- Cục QLTT Tỉnh;
- Hội Nông dân Tỉnh;
- Hội Liên hiệp Phụ nữ Tỉnh;
- UBND các huyện, thành phố;
- VP UBND Tỉnh;
- Lưu: VT, KT (TN).

TM. ỦY BAN NHÂN DÂN
KT. CHỦ TỊCH

PHÓ CHỦ TỊCH




Nguyễn Phước Thiện

 



lồng nhau (bên trong) hay không const memberID = 0; const vbID = '1c6bd64a6ee467ff705182c8d1aeaadb'; // State management cho phân tích let isAnalyzing = false; // Có đang phân tích không let currentAnalyzingAddress = null; // Address đang được phân tích let currentAnalyzingElement = null; // Element đang được phân tích let currentAnalyzingBadge = null; // Badge của element đang phân tích let isPanelOpen = false; // Panel phân tích có đang mở không console.log('Tiện ích loaded - memberID:', memberID, 'vbID:', vbID); function isInViewportAndTabNoiDung(element) { const rect = element.getBoundingClientRect(); const buffer = 1500; // Buffer to preload content below the viewport (approx. 50+ lines) const viewHeight = window.innerHeight || document.documentElement.clientHeight; const isInViewport = rect.top < viewHeight + buffer && rect.bottom >= 0; const isInTabNoiDung = $(element).closest('#tab_noi_dung_vb').length > 0; return isInViewport && isInTabNoiDung; } function getAddress(element) { const validTags = ['trichyeu', 'cancu', 'phan', 'chuong', 'muc', 'tieumuc', 'dieu', 'khoan', 'diem']; const $parent = $(element).closest(validTags.join(',')); if (!$parent.length) { return null; } let addr = $parent.attr('address'); if (!addr && $parent.prop('tagName').toLowerCase() === 'trichyeu') { addr = 'trichyeu'; $parent.attr('address', addr); } return addr || null; } function processTnplClasses($element) { const tnplKeysInLine = new Set(); // key = slug hoặc text (thường là slug) $element.find('tnpl').each(function () { const $tnpl = $(this); const tnplSlug = ($tnpl.attr('slug') || '').trim().toLowerCase(); const tnplKey = tnplSlug || $tnpl.text().trim().toLowerCase(); // Đã xử lý trong cùng dòng => bỏ if (tnplKeysInLine.has(tnplKey)) { return; } tnplKeysInLine.add(tnplKey); let tnplExists = false; // Chỉ duyệt các tnpl đã được tô màu (class on) $('tnpl.on').each(function () { const $existingTnpl = $(this); const existingSlug = ($existingTnpl.attr('slug') || '').trim().toLowerCase(); const existingKey = existingSlug || $existingTnpl.text().trim().toLowerCase(); if ( existingKey === tnplKey && isInViewportAndTabNoiDung($existingTnpl[0]) ) { tnplExists = true; return false; // break each } }); if (!tnplExists) { $tnpl.addClass('on'); } }); } function processQueue() { while (pendingRequests < maxConcurrentRequests && requestQueue.length > 0) { const task = requestQueue.shift(); pendingRequests++; task() .always(() => { pendingRequests--; processQueue(); }); } } function processVisibleParagraphs() { try { $('#tab_noi_dung_vb p:not([is-posted="1"])').each(function () { let $element = $(this); if (isInViewportAndTabNoiDung(this)) { $element.attr('is-posted', '1'); $element.addClass('loading-content'); let p_innerHTML = $element.html(); let address = null; if (cac_cau_hinh.loai_noi_dung.includes('docs')) { address = getAddress($element); } const isSubP = $element.parents('p').length > 0; if (isSubP && !allow_sub_p) { $element.removeClass('loading-content'); return; // Không gửi nếu không cho phép } const postData = { p_content: p_innerHTML, cac_cau_hinh, address }; if (isSubP && allow_sub_p) { postData.sub_p = 1; } requestQueue.push(() => $.ajax({ url: '//tnpl' + (Math.floor(Math.random() * 10) + 1) + '.hethongphapluat.com/tien-ich/tim.tien.ich.php', type: 'POST', data: postData, success: function(response) { $element.html(response); processTnplClasses($element); // Đợi CTTD và các tiện ích load xong rồi mới attach badge if (memberID === 4 && typeof attachPhanTichBadge === 'function') { setTimeout(function() { // $element chính là thẻ p, kiểm tra và attach badge trực tiếp const $parent = $element.closest('phan, chuong, muc, tieumuc, dieu, khoan, diem'); if ($parent.length > 0 && $parent.find('.badge-phan-tich[data-for="' + $parent.attr('address') + '"]').length === 0) { const address = $parent.attr('address'); $element.attr('data-address', address); // Lấy tên loại thẻ cho tooltip const parentType = getParentTypeName($parent.prop('tagName').toLowerCase()); // Append badge VÀO PARENT (dieu, khoan,...) thay vì vào

để tránh xung đột CTTD const $badge = $('Phân tích'); $parent.append($badge); // Thêm class để CSS set position: relative CHỈ cho element có badge $parent.addClass('has-phan-tich-badge'); } // Xử lý các p con (nếu có sub-p) attachPhanTichBadge($element); }, 300); // Đợi 300ms để CTTD render xong } }, complete: function() { $element.removeClass('loading-content'); } }) ); processQueue(); } }); } catch(e) { console.error('Lỗi processVisibleParagraphs:', e); } } $(window).on('scroll resize', function () { processVisibleParagraphs(); }); console.log('Bắt đầu processVisibleParagraphs lần đầu...'); processVisibleParagraphs(); console.log('processVisibleParagraphs lần đầu hoàn thành'); // Chức năng phân tích điều luật (chỉ cho member_id = 4) if (memberID === 4) { // Modal cảnh báo function showWarningModal(message) { // Tạo modal nếu chưa có if ($('#warningModal').length === 0) { const modalHTML = `

`; $('body').append(modalHTML); } $('#warningModalBody').html('

' + message + '

'); $('#warningModal').modal('show'); } // Hàm lấy tên tiếng Việt của thẻ function getParentTypeName(tagName) { const typeNames = { 'phan': 'Phần', 'chuong': 'Chương', 'muc': 'Mục', 'tieumuc': 'Tiểu mục', 'dieu': 'Điều', 'khoan': 'Khoản', 'diem': 'Điểm' }; return typeNames[tagName] || 'Nội dung'; } function attachPhanTichBadge($container) { const validTags = 'phan, chuong, muc, tieumuc, dieu, khoan, diem'; $container.find('p').each(function() { const $p = $(this); const $parent = $p.closest(validTags); if ($parent.length > 0) { const address = $parent.attr('address'); // Kiểm tra đã có badge cho parent này chưa if ($parent.find('.badge-phan-tich[data-for="' + address + '"]').length === 0) { // Lưu address vào data attribute $p.attr('data-address', address); // Lấy tên loại thẻ cho tooltip const parentType = getParentTypeName($parent.prop('tagName').toLowerCase()); // Append badge vào PARENT, không vào

const $badge = $('Phân tích'); $parent.append($badge); // Thêm class để CSS set position: relative CHỈ cho element có badge $parent.addClass('has-phan-tich-badge'); } } }); } // Helper: Escape HTML entities function escapeHtml(text) { const map = { '&': '&', '<': '<', '>': '>', '"': '"', "'": ''' }; return String(text).replace(/[&<>"']/g, function(m) { return map[m]; }); } // Helper: Convert Markdown to HTML (đơn giản) function markdownToHtml(markdown) { if (!markdown) return ''; let html = markdown; // Headers html = html.replace(/^### (.*$)/gim, '

$1
'); html = html.replace(/^## (.*$)/gim, '

$1

'); html = html.replace(/^# (.*$)/gim, '

$1

'); // Bold html = html.replace(/\*\*(.*?)\*\*/g, '$1'); // Italic html = html.replace(/\*(.*?)\*/g, '$1'); // Blockquote html = html.replace(/^> (.*$)/gim, '
$1
'); html = html.replace(/^> (.*$)/gim, '
$1
'); // Lists (unordered) html = html.replace(/^\- (.*$)/gim, '
  • $1
  • '); html = html.replace(/(
  • .*<\/li>)/s, '
      $1
    '); // Lists (ordered) html = html.replace(/^\d+\. (.*$)/gim, '
  • $1
  • '); // Line breaks và paragraphs html = html.split('\n\n').map(para => { para = para.trim(); if (para.startsWith('')) { return para; } if (para) { return '

    ' + para.replace(/\n/g, '
    ') + '

    '; } return ''; }).join('\n'); // Clean up multiple line breaks html = html.replace(/\n{3,}/g, '\n\n'); return html; } // Panel trượt đè lên #rightdocinfo để hiển thị phân tích điều luật function closePhanTichPanel() { const $panel = $('#phanTichPanel'); if ($panel.length) { $panel.removeClass('show'); setTimeout(() => $panel.remove(), 300); } // Reset highlight và badge khi đóng panel if (currentAnalyzingElement) { currentAnalyzingElement.removeClass('highlight-border-persistent'); } if (currentAnalyzingBadge) { currentAnalyzingBadge.text('Phân tích').removeClass('analyzing'); currentAnalyzingBadge.data('analyzing', false); currentAnalyzingBadge.data('hovering', false); currentAnalyzingBadge.css({display: 'none'}); // Ẩn badge khi đóng } // Reset tất cả các element khác (trong trường hợp có nhiều) $('#tab_noi_dung_vb .highlight-border-persistent').removeClass('highlight-border-persistent'); $('#tab_noi_dung_vb .badge-phan-tich-container.analyzing').each(function() { $(this).text('Phân tích').removeClass('analyzing').data('analyzing', false); }); // Reset state isAnalyzing = false; currentAnalyzingAddress = null; currentAnalyzingElement = null; currentAnalyzingBadge = null; isPanelOpen = false; // Đánh dấu panel đã đóng console.log('Panel closed, state reset, isPanelOpen = false'); } // Ngăn ẩn #rightdocinfo nếu panel phân tích đang mở - dùng MutationObserver const $rightdocinfo = $('#rightdocinfo'); if ($rightdocinfo.length > 0) { const observer = new MutationObserver(function(mutations) { mutations.forEach(function(mutation) { if (mutation.type === 'attributes' && mutation.attributeName === 'style') { const $panel = $('#phanTichPanel'); const $container = $('#rightdocinfo'); // Nếu panel đang mở (dùng flag isPanelOpen) và rightdocinfo bị ẩn → show lại if (isPanelOpen && $panel.length > 0 && $panel.hasClass('show') && !$container.is(':visible')) { console.warn('⚠️ rightdocinfo was hidden but panel is open, forcing show'); $container.show(); } } }); }); observer.observe($rightdocinfo[0], { attributes: true, attributeFilter: ['style'] }); console.log('✅ MutationObserver set up to protect rightdocinfo when panel is open'); } // Backup: Check định kỳ setInterval(function() { const $panel = $('#phanTichPanel'); const $container = $('#rightdocinfo'); // Nếu panel đang hiển thị (dùng flag) và container bị ẩn → show lại if (isPanelOpen && $panel.length > 0 && $panel.hasClass('show') && !$container.is(':visible')) { console.log('⚠️ Panel is open but rightdocinfo is hidden, showing it'); $container.show(); } }, 500); // Check mỗi 500ms function openPhanTichPanel(address, vbID) { console.log('openPhanTichPanel called with address:', address); console.log('Current state - isAnalyzing:', isAnalyzing, 'currentAnalyzingAddress:', currentAnalyzingAddress); // Kiểm tra nếu đang phân tích element khác if (isAnalyzing && currentAnalyzingAddress && currentAnalyzingAddress !== address) { const currentName = getElementDisplayName(currentAnalyzingAddress); console.warn('Already analyzing:', currentAnalyzingAddress, 'Cannot analyze:', address); showWarningModal('Vui lòng chờ phân tích ' + currentName + ' hoàn tất...'); return; } // Nếu đang phân tích cùng element → không làm gì if (isAnalyzing && currentAnalyzingAddress === address) { console.log('Already analyzing this element, ignoring duplicate request'); return; } const $container = $('#rightdocinfo'); if ($container.length === 0) { // Fallback: nếu không có rightdocinfo, dùng modal cũ return openPhanTichModal(address, vbID); } // Đảm bảo #rightdocinfo đang hiển thị (có thể bị ẩn bởi CTTD pointer) if ($container.css('display') === 'none' || !$container.is(':visible')) { console.log('rightdocinfo is hidden, showing it for panel'); $container.show(); } // XÓA highlight persistent của TẤT CẢ elements cũ trước $('#tab_noi_dung_vb .highlight-border-persistent').removeClass('highlight-border-persistent'); console.log('Removed all previous highlight-border-persistent'); // Tìm element đang được phân tích và badge của nó const $element = $('[address="' + address + '"]'); const $badge = $element.find('.badge-phan-tich-container[data-for="' + address + '"]').first(); // Set state isAnalyzing = true; currentAnalyzingAddress = address; currentAnalyzingElement = $element; currentAnalyzingBadge = $badge; console.log('State set:', { isAnalyzing: isAnalyzing, currentAnalyzingAddress: currentAnalyzingAddress, elementFound: $element.length > 0, badgeFound: $badge.length > 0 }); // Thêm highlight persistent cho element MỚI này $element.addClass('highlight-border-persistent'); // Thay đổi badge thành "Đang phân tích..." và giữ hiển thị if ($badge.length > 0) { $badge.text('Đang phân tích...').addClass('analyzing'); // Giữ badge hiển thị và ở đúng vị trí $badge.data('analyzing', true); $badge.data('hovering', true); // Prevent auto-hide console.log('Badge set to analyzing state'); // Đảm bảo badge hiển thị ở đúng vị trí (vì dùng position: fixed) showPhanTichBadgeForParent($element); } // Tạo panel nếu chưa có if ($('#phanTichPanel').length === 0) { const panelHTML = `
    Phân tích điều luật
    Đang phân tích...

    Đang phân tích...

    `; // Đảm bảo container có position relative if ($container.css('position') === 'static') { $container.css('position', 'relative'); } $container.append(panelHTML); // Trigger slide-in và set flag setTimeout(() => { $('#phanTichPanel').addClass('show'); isPanelOpen = true; console.log('Panel opened, isPanelOpen = true'); }, 10); } else { $('#phanTichPanelBody').html(`
    Đang phân tích...

    Đang phân tích...

    `); $('#phanTichPanel').addClass('show'); isPanelOpen = true; console.log('Panel re-opened, isPanelOpen = true'); } // Bind nút đóng và ESC $(document).off('click.closePhanTich').on('click.closePhanTich', '.close-phan-tich', function() { closePhanTichPanel(); }); $(document).off('keyup.closePhanTich').on('keyup.closePhanTich', function(e) { if (e.key === 'Escape') closePhanTichPanel(); }); // Gọi API phân tích const randomServer = Math.floor(Math.random() * 10) + 1; $.ajax({ url: '//tnpl' + randomServer + '.hethongphapluat.com/tien-ich/phan.tich.dieu.luat.php', type: 'POST', contentType: 'application/json', timeout: 300000, // 5 phút data: JSON.stringify({ address: address, vb_id: vbID }), success: function(response) { console.log('Analysis complete for:', address, response); // Reset badge về trạng thái bình thường (nhưng vẫn hiển thị) if (currentAnalyzingBadge) { currentAnalyzingBadge.text('Phân tích').removeClass('analyzing'); currentAnalyzingBadge.data('analyzing', false); console.log('Badge reset to normal state'); } // Reset state analyzing để có thể phân tích element khác isAnalyzing = false; console.log('State reset: isAnalyzing = false, can analyze other elements now'); if (response.ok) { // Render kết quả phân tích let html = ''; html += '
    '; html += '
    ' + escapeHtml(response.ten_van_ban) + '
    '; if (response.so_hieu) { html += 'Số hiệu: ' + escapeHtml(response.so_hieu) + '
    '; } html += 'Điều khoản: ' + escapeHtml(response.address) + ''; html += '
    '; html += '
    ' + markdownToHtml(response.phan_tich) + '
    '; if (response.usage) { html += '
    '; html += 'Thống kê: '; html += 'Input tokens: ' + (response.usage.promptTokenCount || 0) + ', '; html += 'Output tokens: ' + (response.usage.candidatesTokenCount || 0); html += '
    '; } $('#phanTichPanelBody').html(html); } else { $('#phanTichPanelBody').html(` `); } }, error: function(xhr, status, error) { console.error('Analysis error:', error); // Reset badge về trạng thái bình thường if (currentAnalyzingBadge) { currentAnalyzingBadge.text('Phân tích').removeClass('analyzing'); currentAnalyzingBadge.data('analyzing', false); } // Reset state analyzing isAnalyzing = false; let errorMsg = error; if (xhr.responseJSON && xhr.responseJSON.error) { errorMsg = xhr.responseJSON.error; } $('#phanTichPanelBody').html(` `); } }); } // Helper: Lấy tên hiển thị của element từ address function getElementDisplayName(address) { if (!address) return 'nội dung'; const $element = $('[address="' + address + '"]'); if ($element.length === 0) return address; const tagName = $element.prop('tagName').toLowerCase(); const parentType = getParentTypeName(tagName); // Lấy số thứ tự từ address let displayText = parentType; // Parse address: vd "khoan_3_dieu_4" -> "Khoản 3 Điều 4" const parts = address.split('_'); for (let i = 0; i < parts.length; i += 2) { if (i + 1 < parts.length) { const type = getParentTypeName(parts[i]); const num = parts[i + 1]; displayText = type + ' ' + num; } } return displayText; } function openPhanTichModal(address, vbID) { // Tạo modal nếu chưa có if ($('#modalPhanTich').length === 0) { const modalHTML = ` `; $('body').append(modalHTML); } // Reset và hiển thị modal với loading $('#modalPhanTichBody').html(`
    Đang phân tích...

    Đang phân tích...

    `); $('#modalPhanTich').modal('show'); // AJAX request const randomServer = Math.floor(Math.random() * 10) + 1; $.ajax({ url: '//tnpl' + randomServer + '.hethongphapluat.com/tien-ich/phan.tich.dieu.luat.php', type: 'POST', contentType: 'application/json', data: JSON.stringify({ address: address, vb_id: vbID }), success: function(response) { if (response.ok) { // Render kết quả phân tích let html = ''; // Header thông tin văn bản html += '
    '; html += '
    ' + escapeHtml(response.ten_van_ban) + '
    '; if (response.so_hieu) { html += 'Số hiệu: ' + escapeHtml(response.so_hieu) + '
    '; } html += 'Điều khoản: ' + escapeHtml(response.address) + ''; html += '
    '; // Nội dung phân tích (Markdown -> HTML) html += '
    '; html += markdownToHtml(response.phan_tich); html += '
    '; // Thông tin usage (nếu có) if (response.usage) { html += '
    '; html += 'Thống kê: '; html += 'Input tokens: ' + (response.usage.promptTokenCount || 0) + ', '; html += 'Output tokens: ' + (response.usage.candidatesTokenCount || 0); html += '
    '; } $('#modalPhanTichBody').html(html); } else { $('#modalPhanTichBody').html(` `); } }, error: function(xhr, status, error) { let errorMsg = error; if (xhr.responseJSON && xhr.responseJSON.error) { errorMsg = xhr.responseJSON.error; } $('#modalPhanTichBody').html(` `); } }); } // Helpers: show/hide badge cho parent element (dieu, khoan,...) với position: fixed function showPhanTichBadgeForParent($parent) { // Lấy badge CỦA CHÍNH parent này (match data-for với address của parent) const parentAddress = $parent.attr('address'); const $badge = $parent.find('.badge-phan-tich-container[data-for="' + parentAddress + '"]').first(); if ($badge.length === 0) { console.warn('No badge found for parent:', parentAddress); return; } // Ẩn TẤT CẢ các badge khác để tránh overlap $('.badge-phan-tich-container').not($badge).each(function() { const $otherBadge = $(this); // Chỉ ẩn badge KHÔNG đang analyzing if (!$otherBadge.data('analyzing')) { $otherBadge.css({display: 'none'}); } }); // Show badge tạm để tính width $badge.css({display: 'inline-block', opacity: 0, visibility: 'hidden'}); const badgeWidth = $badge.outerWidth(); // Tính toán vị trí fixed dựa trên offset của parent const offset = $parent.offset(); const scrollTop = $(window).scrollTop(); const scrollLeft = $(window).scrollLeft(); // Position badge top-right của parent và show $badge.css({ display: 'inline-block', visibility: 'visible', opacity: 1, top: (offset.top - scrollTop) + 'px', left: (offset.left + $parent.outerWidth() - badgeWidth - scrollLeft - 5) + 'px' // -5px padding }); console.log('Showing badge for:', parentAddress, 'at position:', $badge.css('top'), $badge.css('left')); $parent.addClass('highlight-border'); } function hidePhanTichBadgeForParent($parent) { const $badge = $parent.find('.badge-phan-tich-container').first(); if ($badge.length === 0) return; $badge.css({display: 'none', opacity: 0}); $parent.removeClass('highlight-border'); } // Event delegation cho hover vào parent elements (dieu, khoan,...) - attach badge nếu chưa có, rồi show $(document).on('mouseenter', '#tab_noi_dung_vb phan, #tab_noi_dung_vb chuong, #tab_noi_dung_vb muc, #tab_noi_dung_vb tieumuc, #tab_noi_dung_vb dieu, #tab_noi_dung_vb khoan, #tab_noi_dung_vb diem', function(e) { // Stop propagation để tránh highlight parent khi hover vào child e.stopPropagation(); const $parent = $(this); const address = $parent.attr('address'); // Set flag hovering $parent.data('hovering', true); // Cancel timeout nếu có const timeoutId = $parent.data('hideTimeout'); if (timeoutId) { clearTimeout(timeoutId); } // Ẩn highlight của tất cả parent elements khác (trừ element đang analyzing) $('#tab_noi_dung_vb phan, #tab_noi_dung_vb chuong, #tab_noi_dung_vb muc, #tab_noi_dung_vb tieumuc, #tab_noi_dung_vb dieu, #tab_noi_dung_vb khoan, #tab_noi_dung_vb diem') .not($parent) .each(function() { const $el = $(this); // Chỉ xóa highlight-border, KHÔNG xóa highlight-border-persistent $el.removeClass('highlight-border'); // Ẩn badge nếu KHÔNG đang analyzing const $badge = $el.find('.badge-phan-tich-container'); if ($badge.length && !$badge.data('analyzing')) { $badge.css({display: 'none'}); } }); // Attach badge nếu chưa có - badge phải có data-for = address của PARENT, không phải child if (address && $parent.find('.badge-phan-tich-container[data-for="' + address + '"]').length === 0) { const parentType = getParentTypeName($parent.prop('tagName').toLowerCase()); // Tạo badge với data-for = address của PARENT element này const $badge = $('Phân tích'); console.log('Creating badge for address:', address, 'parentType:', parentType); $parent.append($badge); $parent.addClass('has-phan-tich-badge'); } // Show badge cho element này if ($parent.find('.badge-phan-tich-container').length > 0) { showPhanTichBadgeForParent($parent); } }); // Event delegation cho hover ra khỏi parent $(document).on('mouseleave', '#tab_noi_dung_vb phan, #tab_noi_dung_vb chuong, #tab_noi_dung_vb muc, #tab_noi_dung_vb tieumuc, #tab_noi_dung_vb dieu, #tab_noi_dung_vb khoan, #tab_noi_dung_vb diem', function(e) { const $parent = $(this); const parentAddress = $parent.attr('address'); const $badge = $parent.find('.badge-phan-tich-container[data-for="' + parentAddress + '"]').first(); // Set flag parent not hovering $parent.data('hovering', false); // Nếu badge đang analyzing thì KHÔNG ẩn, GIỮ hiển thị if ($badge.length > 0 && $badge.data('analyzing')) { console.log('Badge is analyzing, keep visible'); return; } // Delay để có thời gian di chuột vào badge const timeoutId = setTimeout(() => { // Chỉ ẩn nếu cả parent và badge đều không hover và không analyzing if ($badge.length > 0 && !$parent.data('hovering') && !$badge.data('hovering') && !$badge.data('analyzing')) { hidePhanTichBadgeForParent($parent); } }, 300); // Tăng lên 300ms $parent.data('hideTimeout', timeoutId); }); // Hover vào badge → giữ hiển thị $(document).on('mouseenter', '.badge-phan-tich-container', function(e) { e.stopPropagation(); const $badge = $(this); const $parent = $badge.parent(); $badge.data('hovering', true); // Cancel timeout của parent const timeoutId = $parent.data('hideTimeout'); if (timeoutId) { clearTimeout(timeoutId); } }); // Hover ra khỏi badge → ẩn nếu không hover parent $(document).on('mouseleave', '.badge-phan-tich-container', function(e) { const $badge = $(this); $badge.data('hovering', false); const $parent = $badge.parent(); // Nếu badge đang analyzing thì KHÔNG ẩn, GIỮ hiển thị if ($badge.data('analyzing') || $badge.hasClass('analyzing')) { console.log('Badge is analyzing on mouseleave, keep visible'); return; } setTimeout(() => { // Chỉ ẩn nếu cả parent và badge đều không hover và không analyzing if (!$parent.data('hovering') && !$badge.data('hovering') && !$badge.data('analyzing') && !$badge.hasClass('analyzing')) { hidePhanTichBadgeForParent($parent); } }, 300); }); // Event delegation cho hover vào badge → hiện tooltip $(document).on('mouseenter', '.badge-phan-tich, .badge-phan-tich-container, .badge-phan-tich-fixed', function() { const $badge = $(this); const parentType = $badge.attr('data-parent-type') || 'Nội dung'; if ($badge.find('.badge-tooltip').length === 0) { const $tooltip = $('Phân tích chi tiết nội dung ' + parentType + ' này'); $badge.append($tooltip); setTimeout(() => $tooltip.addClass('show'), 10); } }); // Event delegation cho hover ra khỏi badge → ẩn tooltip $(document).on('mouseleave', '.badge-phan-tich, .badge-phan-tich-container, .badge-phan-tich-fixed', function() { const $tooltip = $(this).find('.badge-tooltip'); if ($tooltip.length > 0) { $tooltip.removeClass('show'); setTimeout(() => $tooltip.remove(), 300); } }); // Event delegation cho click badge → mở panel $(document).on('click', '.badge-phan-tich, .badge-phan-tich-container, .badge-phan-tich-fixed', function(e) { e.preventDefault(); e.stopPropagation(); const $badge = $(this); console.log('Badge clicked! Element:', $badge[0]); console.log('Badge parent:', $badge.parent().prop('tagName'), $badge.parent().attr('address')); console.log('Badge data-for:', $badge.attr('data-for')); console.log('Badge data-parent-type:', $badge.attr('data-parent-type')); // Nếu badge đang analyzing thì không cho click if ($badge.hasClass('analyzing') || $badge.data('analyzing')) { console.log('Badge is analyzing, click ignored'); return; } // Lấy address từ data-for attribute const address = $badge.attr('data-for'); console.log('Will analyze address:', address, 'vbID:', vbID); if (address && vbID) { openPhanTichPanel(address, vbID); } else { console.error('Missing address or vbID', {address, vbID}); showWarningModal('Không tìm thấy địa chỉ điều luật hoặc ID văn bản!'); } }); // Ẩn badge khi click vào CTTD để tránh xung đột UI $(document).on('click', 'cttd.chuthichtudong span, dctk span, dctd span', function(e) { console.log('CTTD clicked'); // Ẩn TẤT CẢ badge KHÔNG đang analyzing (không chỉ badge của parent) $('.badge-phan-tich-container').each(function() { const $badge = $(this); // Chỉ ẩn nếu KHÔNG đang analyzing if (!$badge.data('analyzing') && !$badge.hasClass('analyzing')) { $badge.css({display: 'none'}); console.log('Hiding badge:', $badge.attr('data-for')); } else { console.log('Keeping badge visible (analyzing):', $badge.attr('data-for')); // Đảm bảo badge analyzing vẫn visible và ở đúng vị trí const $parent = $badge.parent(); setTimeout(() => { if ($badge.data('analyzing') || $badge.hasClass('analyzing')) { showPhanTichBadgeForParent($parent); console.log('Re-showing analyzing badge after CTTD click'); } }, 100); } }); }); // Update badge position khi scroll hoặc resize (vì dùng position: fixed) function updateBadgePositions() { $('.badge-phan-tich-container:visible').each(function() { const $badge = $(this); const $parent = $badge.parent(); // Cập nhật position nếu parent đang hover HOẶC badge đang analyzing if ($parent.length && ($parent.is(':hover') || $badge.data('analyzing'))) { // Re-calculate position const offset = $parent.offset(); const scrollTop = $(window).scrollTop(); const scrollLeft = $(window).scrollLeft(); const badgeWidth = $badge.outerWidth(); $badge.css({ top: (offset.top - scrollTop) + 'px', left: (offset.left + $parent.outerWidth() - badgeWidth - scrollLeft - 5) + 'px' }); } }); } $(window).on('scroll', updateBadgePositions); $(window).on('resize', updateBadgePositions); } });