From c44e67d80ce3b000568017c0df4e5a7f436cf4a9 Mon Sep 17 00:00:00 2001 From: akib22 Date: Wed, 8 Oct 2025 19:59:24 +0600 Subject: [PATCH 1/8] Add missing Bengali translations - ordered the translation keys same as en.js file to check the difference easily - improve word selection and sentence structure Issue: #627 --- src/i18n/locales/bn.js | 169 +++++++++++++++++++++++++---------------- 1 file changed, 104 insertions(+), 65 deletions(-) diff --git a/src/i18n/locales/bn.js b/src/i18n/locales/bn.js index 740f83384..468acb4b0 100644 --- a/src/i18n/locales/bn.js +++ b/src/i18n/locales/bn.js @@ -7,12 +7,17 @@ const bengali = { const bn = { translation: { report_bug: "বাগ রিপোর্ট করুন", - import_from: "ইম্পোর্ট করুন", import: "ইম্পোর্ট করুন", + inherits: "উত্তরাধিকার", + merging_column_w_inherited_definition: + "টেবিল '{{tableName}}'-এর '{{fieldName}}' কলাম উত্তরাধিকারসূত্রে একীভূত হবে", + import_from: "ইম্পোর্ট করুন", file: "ফাইল", new: "নতুন", new_window: "নতুন উইন্ডো", + no_saved_diagrams: "আপনার কোনো সেভ করা ডায়াগ্রাম নেই", open: "ওপেন করুন", + open_recent: "সাম্প্রতিক ফাইল খুলুন", save: "সেভ করুন", save_as: "নতুন নামে সেভ করুন", save_as_template: "টেমপ্লেট হিসাবে সেভ করুন", @@ -20,24 +25,23 @@ const bn = { rename: "নাম পরিবর্তন করুন", delete_diagram: "ডায়াগ্রাম ডিলিট করুন", are_you_sure_delete_diagram: - "আপনি কি এই ডায়াগ্রামটি মুছে ফেলতে চান? এই অপারেশনটি অপরিবর্তনীয়।", + "আপনি কি নিশ্চিত যে আপনি ডায়াগ্রাম মুছে ফেলবেন? একবার মুছে ফেললে আর ফেরত আনা যাবে না।", oops_smth_went_wrong: "ওহো! কিছু ভুল হয়েছে।", import_diagram: "ডায়াগ্রাম ইম্পোর্ট করুন", import_from_source: "SQL থেকে ইম্পোর্ট করুন", export_as: "রপ্তানি হিসাবে", - export_source: "SQL রপ্তানি করুন", + export_source: "SQL এক্সপোর্ট করুন", models: "মডেল", exit: "বেরিয়ে যান", edit: "এডিট করুন", undo: "পূর্বাবস্থা ফিরিয়ে নিন", redo: "পুনরায় করুন", clear: "মুছে ফেলুন", - are_you_sure_clear: - "আপনি কি ডায়াগ্রামটি মুছে ফেলতে চান? এটি অপরিবর্তনীয়।", + are_you_sure_clear: "আপনি কি নিশ্চিত যে আপনি ডায়াগ্রাম মুছে ফেলবেন? একবার মুছে ফেললে আর ফেরত আনা যাবে না।", cut: "কাট করুন", copy: "কপি করুন", paste: "পেস্ট করুন", - duplicate: "প্রতিলিপি করুন", + duplicate: "ডুপ্লিকেট করুন", delete: "মুছে ফেলুন", copy_as_image: "ছবি হিসাবে কপি করুন", view: "ভিউ", @@ -46,9 +50,11 @@ const bn = { issues: "সমস্যা", presentation_mode: "প্রেজেন্টেশন মোড", strict_mode: "স্ট্রিক্ট মোড", - field_details: "ক্ষেত্রের বিস্তারিত", + field_details: "ফিল্ডের বিবরণ", reset_view: "ভিউ রিসেট করুন", show_grid: "গ্রিড দেখান", + snap_to_grid: "গ্রিডে স্ন্যাপ করুন", + show_datatype: "ডেটা টাইপ দেখান", show_cardinality: "কার্ডিনালিটি দেখান", theme: "থিম", light: "লাইট", @@ -70,18 +76,17 @@ const bn = { table_width: "টেবিলের প্রস্থ", language: "ভাষা", flush_storage: "স্টোরেজ ফ্লাশ করুন", - are_you_sure_flush_storage: - "আপনি কি স্টোরেজ ফ্লাশ করতে চান? এটি আপনার সমস্ত ডায়াগ্রাম এবং কাস্টম টেমপ্লেটগুলি মুছে ফেলবে।", + are_you_sure_flush_storage: "আপনি কি নিশ্চিত যে আপনি স্টোরেজ ফ্লাশ করতে চান? এই পদক্ষেপের ফলে আপনার সকল ডায়াগ্রাম ও কাস্টম টেমপ্লেট স্থায়ীভাবে মুছে যাবে এবং আর ফিরে পাওয়া যাবে না।", storage_flushed: "স্টোরেজ ফ্লাশ হয়েছে", help: "সাহায্য", shortcuts: "শর্টকাট", ask_on_discord: "ডিসকর্ডে আমাদের জিজ্ঞাসা করুন", - feedback: "প্রতিক্রিয়া", + feedback: "ফিডব্যাক দিন", no_changes: "কোনও পরিবর্তন নেই", loading: "লোড হচ্ছে...", last_saved: "শেষ সেভ", saving: "সেভ হচ্ছে...", - failed_to_save: "সেভ ব্যর্থ হয়েছে", + failed_to_save: "সেভ করতে ব্যর্থ হয়েছে", fit_window_reset: "উইন্ডোতে ফিট করুন / রিসেট করুন", zoom: "জুম", add_table: "টেবিল যোগ করুন", @@ -90,38 +95,38 @@ const bn = { add_type: "টাইপ যোগ করুন", to_do: "টু-ডু", tables: "টেবিল", - relationships: "সম্পর্কগুলি", - subject_areas: "বিষয় এলাকা", + relationships: "রিলেশনশিপগুলি", + subject_areas: "সাবজেক্ট এরিয়া", notes: "নোট", - types: "প্রকার", + types: "টাইপ", search: "অনুসন্ধান করুন...", no_tables: "কোনও টেবিল নেই", no_tables_text: "আপনার ডায়াগ্রামটি তৈরি করা শুরু করুন!", - no_relationships: "কোনও সম্পর্ক নেই", - no_relationships_text: - "ক্ষেত্রগুলিকে সংযুক্ত করতে এবং সম্পর্ক গঠনের জন্য টানুন!", - no_subject_areas: "কোনও বিষয় এলাকা নেই", - no_subject_areas_text: "টেবিলগুলি গোষ্ঠীবদ্ধ করতে বিষয় এলাকা যোগ করুন!", + no_relationships: "কোনও রিলেশন নেই", + no_relationships_text: "ফিল্ডগুলো টেনে একে অপরের সঙ্গে সংযুক্ত করুন এবং রিলেশন তৈরি করুন!", + no_subject_areas: "কোনও সাবজেক্ট এরিয়া নেই", + no_subject_areas_text: "টেবিলগুলি গ্রুপ করতে সাবজেক্ট এরিয়া যোগ করুন!", no_notes: "কোনও নোট নেই", no_notes_text: "অতিরিক্ত তথ্য রেকর্ড করার জন্য নোট ব্যবহার করুন", - no_types: "কোনও প্রকার নেই", + no_types: "কোনও টাইপ নেই", no_types_text: "আপনার নিজস্ব কাস্টম ডেটা টাইপগুলি তৈরি করুন", - no_issues: "কোনও সমস্যা সনাক্ত করা হয়নি।", + no_issues: "কোনও সমস্যা সনাক্ত করা যায়নি।", strict_mode_is_on_no_issues: "স্ট্রিক্ট মোড বন্ধ রয়েছে, তাই কোনও সমস্যা প্রদর্শিত হবে না।", name: "নাম", - type: "প্রকার", + type: "টাইপ", null: "নাল", not_null: "নাল নয়", + nullable: "নালযোগ্য", primary: "প্রাথমিক", - unique: "অনন্য", - autoincrement: "স্বয়ংক্রিয় বৃদ্ধি", - default_value: "ডিফল্ট মান", + unique: "ইউনিক", + autoincrement: "অটোমেটিক বৃদ্ধি পাবে", + default_value: "ডিফল্ট ভ্যালু", check: "চেক এক্সপ্রেশন", this_will_appear_as_is: "*এটি তৈরি করা স্ক্রিপ্টে অপরিবর্তিত অবস্থায় প্রদর্শিত হবে।", comment: "মন্তব্য", - add_field: "ক্ষেত্র যোগ করুন", + add_field: "ফিল্ড যোগ করুন", values: "মান", size: "আকার", precision: "প্রেসিশন", @@ -129,33 +134,33 @@ const bn = { use_for_batch_input: "ব্যাচ ইনপুটের জন্য ব্যবহার করুন", indices: "ইনডিসেস", add_index: "ইনডেক্স যোগ করুন", - select_fields: "ক্ষেত্রগুলি নির্বাচন করুন", + select_fields: "ফিল্ড নির্বাচন করুন", title: "শিরোনাম", not_set: "সেট করা হয়নি", - foreign: "বৈদেশিক", + foreign: "ফরেন", cardinality: "কার্ডিনালিটি", on_update: "আপডেটের সময়", on_delete: "ডিলিটের সময়", swap: "সোয়াপ", - one_to_one: "এক থেকে এক", - one_to_many: "এক থেকে অনেক", - many_to_one: "অনেক থেকে এক", + one_to_one: "ওয়ান টু ওয়ান", + one_to_many: "ওয়ান টু ম্যানি", + many_to_one: "ম্যানি টু ওয়ান", content: "বিষয়বস্তু", types_info: "এই বৈশিষ্ট্যটি PostgreSQL-এর মত অবজেক্ট-রিলেশনাল DBMS-এর জন্য।\nযদি MySQL বা MariaDB এর জন্য ব্যবহার করা হয় তবে একটি JSON টাইপ তৈরি হবে সংশ্লিষ্ট json বৈধতা যাচাই সহ।\nযদি SQLite এর জন্য ব্যবহার করা হয় তবে এটি একটি BLOB এ অনুবাদ হবে।\nযদি MSSQL এর জন্য ব্যবহার করা হয় তবে প্রথম ক্ষেত্রের একটি টাইপ এলিয়াস তৈরি হবে।", table_deleted: "টেবিল মুছে ফেলা হয়েছে", area_deleted: "এরিয়া মুছে ফেলা হয়েছে", note_deleted: "নোট মুছে ফেলা হয়েছে", - relationship_deleted: "সম্পর্ক মুছে ফেলা হয়েছে", + relationship_deleted: "রিলেশন মুছে ফেলা হয়েছে", type_deleted: "টাইপ মুছে ফেলা হয়েছে", - cannot_connect: "সংযোগ করা যাচ্ছে না, কলামগুলির বিভিন্ন প্রকার আছে", + cannot_connect: "সংযোগ করা যাচ্ছে না, কলামগুলির টাইপ ভিন্ন", copied_to_clipboard: "ক্লিপবোর্ডে কপি করা হয়েছে", create_new_diagram: "নতুন ডায়াগ্রাম তৈরি করুন", cancel: "বাতিল করুন", open_diagram: "ডায়াগ্রাম ওপেন করুন", rename_diagram: "ডায়াগ্রামের নাম পরিবর্তন করুন", - export: "রপ্তানি করুন", - export_image: "ছবি রপ্তানি করুন", + export: "এক্সপোর্ট করুন", + export_image: "ছবি এক্সপোর্ট করুন", create: "তৈরি করুন", confirm: "নিশ্চিত করুন", last_modified: "শেষ সংশোধন", @@ -165,36 +170,36 @@ const bn = { "আপনার টেবিল এবং কলামগুলি স্বয়ংক্রিয়ভাবে তৈরি করতে একটি SQL ফাইল আপলোড করুন।", overwrite_existing_diagram: "বিদ্যমান ডায়াগ্রামটি ওভাররাইট করুন", only_mysql_supported: - "*এখন পর্যন্ত শুধুমাত্র MySQL স্ক্রিপ্ট লোডিং সমর্থিত।", + "*এখন পর্যন্ত শুধুমাত্র MySQL স্ক্রিপ্ট সাপোর্ট করে।", blank: "খালি", filename: "ফাইলের নাম", table_w_no_name: "নাম ছাড়াই একটি টেবিল ঘোষণা করা হয়েছে", - duplicate_table_by_name: "'{{tableName}}' নামকরণ করা টেবিলের অনুলিপি", - empty_field_name: "'{{tableName}}' টেবিলে ফাঁকা ক্ষেত্রের `name`", - empty_field_type: "'{{tableName}}' টেবিলে ফাঁকা ক্ষেত্রের `type`", + duplicate_table_by_name: "'{{tableName}}' নামের টেবিল ডুপ্লিকেট হয়েছে।", + empty_field_name: "টেবিল '{{tableName}}'-এ `name` ফিল্ডটি খালি আছে", + empty_field_type: "টেবিল '{{tableName}}'-এ `type` ফিল্ডটি খালি আছে", no_values_for_field: - "'{{tableName}}' টেবিলের '{{fieldName}}' ক্ষেত্রটি `{{type}}` প্রকারের, তবে কোনও মান নির্দিষ্ট করা হয়নি", + "টেবিল '{{tableName}}'-এর '{{fieldName}}' ফিল্ডটি `{{type}}` টাইপের, তবে কোনও ভ্যালু প্রদান করা হয়নি", default_doesnt_match_type: - "'{{tableName}}' টেবিলের '{{fieldName}}' ক্ষেত্রটির জন্য ডিফল্ট মান তার প্রকারের সাথে মেলে না", + "টেবিল '{{tableName}}'-এর '{{fieldName}}' ফিল্ডটির জন্য ডিফল্ট ভ্যালু তার টাইপের সাথে মেলে না", not_null_is_null: - "'{{tableName}}' টেবিলের '{{fieldName}}' ক্ষেত্রটি NOT NULL তবে ডিফল্ট NULL", + "টেবিল '{{tableName}}'-এর '{{fieldName}}' ফিল্ডটি NOT NULL তবে ডিফল্ট ভ্যালু NULL", duplicate_fields: - "'{{tableName}}' টেবিলের '{{fieldName}}' নামে ডুপ্লিকেট ক্ষেত্র", + "টেবিল '{{tableName}}'-এর '{{fieldName}}' নামে ডুপ্লিকেট ফিল্ড আছে", duplicate_index: - "'{{tableName}}' টেবিলের '{{indexName}}' নামে ডুপ্লিকেট ইনডেক্স", - empty_index: "'{{tableName}}' টেবিলের ইনডেক্স কোনও কলামকে ইনডেক্স করে না", - no_primary_key: "'{{tableName}}' টেবিলের কোনও প্রাথমিক কী নেই", - type_with_no_name: "নাম ছাড়াই একটি টাইপ ঘোষণা করা হয়েছে", + "টেবিল '{{tableName}}'-এর '{{indexName}}' নামে ডুপ্লিকেট ইনডেক্স আছে", + empty_index: "টেবিল '{{tableName}}'-এর ইনডেক্স কোনও কলামকে ইনডেক্স করে না", + no_primary_key: "টেবিল '{{tableName}}'-এর কোনও প্রাইমারি কী নেই", + type_with_no_name: "নাম ছাড়াই একটি টাইপ ডিক্লেয়ার করা হয়েছে", duplicate_types: "'{{typeName}}' নামকরণের সাথে ডুপ্লিকেট টাইপ", - type_w_no_fields: "ক্ষেত্র ছাড়াই '{{typeName}}' টাইপ ঘোষণা করা হয়েছে", - empty_type_field_name: "'{{typeName}}' টাইপের ফাঁকা ক্ষেত্রের `name`", - empty_type_field_type: "'{{typeName}}' টাইপের ফাঁকা ক্ষেত্রের `type`", + type_w_no_fields: "ফিল্ড ছাড়াই '{{typeName}}' টাইপ ডিক্লেয়ার করা হয়েছে", + empty_type_field_name: "টাইপ '{{typeName}}'-এ `name` ফিল্ডটি খালি আছে", + empty_type_field_type: "টাইপ '{{typeName}}'-এ `type` ফিল্ডটি খালি আছে", no_values_for_type_field: - "'{{typeName}}' টাইপের '{{fieldName}}' ক্ষেত্রটি `{{type}}` প্রকারের, তবে কোনও মান নির্দিষ্ট করা হয়নি", + "টাইপ '{{typeName}}'-এর '{{fieldName}}' ফিল্ডটি `{{type}}` টাইপের, তবে কোনও ভ্যালু প্রদান করা হয়নি", duplicate_type_fields: - "'{{typeName}}' টাইপের '{{fieldName}}' নামে ডুপ্লিকেট ক্ষেত্র", - duplicate_reference: "'{{refName}}' নামে ডুপ্লিকেট রেফারেন্স", - circular_dependency: "'{{refName}}' টেবিল জড়িত একটি চক্রাকার নির্ভরতা", + "টাইপ '{{typeName}}'-এর '{{fieldName}}' নামে ডুপ্লিকেট ফিল্ড আছে", + duplicate_reference: "'{{refName}}' নামে ডুপ্লিকেট রেফারেন্স আছে", + circular_dependency: "টেবিল '{{refName}}'-এ সার্কুলার ডিপেন্ডেন্সি তৈরি হয়েছে", timeline: "টাইমলাইন", priority: "অগ্রাধিকার", none: "কোনও নয়", @@ -210,28 +215,27 @@ const bn = { no_tasks: "আপনার এখনও কোনও কাজ নেই।", no_activity: "আপনার এখনও কোনও কার্যকলাপ নেই।", move_element: "{{name}} কে {{coords}} তে সরান", - edit_area: "{{extra}} এরিয়া {{areaName}} সম্পাদনা করুন", + edit_area: "{{extra}} এরিয়া {{areaName}} এডিট করুন", delete_area: "এরিয়া {{areaName}} মুছুন", - edit_note: "{{extra}} নোট {{noteTitle}} সম্পাদনা করুন", + edit_note: "{{extra}} নোট {{noteTitle}} এডিট করুন", delete_note: "নোট {{noteTitle}} মুছুন", - edit_table: "{{extra}} টেবিল {{tableName}} সম্পাদনা করুন", + edit_table: "{{extra}} টেবিল {{tableName}} এডিট করুন", delete_table: "টেবিল {{tableName}} মুছুন", - edit_type: "{{extra}} টাইপ {{typeName}} সম্পাদনা করুন", + edit_type: "{{extra}} টাইপ {{typeName}} এডিট করুন", delete_type: "টাইপ {{typeName}} মুছুন", - add_relationship: "সম্পর্ক যোগ করুন", - edit_relationship: "{{extra}} সম্পর্ক {{refName}} সম্পাদনা করুন", - delete_relationship: "সম্পর্ক {{refName}} মুছুন", + add_relationship: "রিলেশন যোগ করুন", + edit_relationship: "{{extra}} রিলেশন {{refName}} এডিট করুন", + delete_relationship: "রিলেশন {{refName}} মুছুন", not_found: "খুঁজে পাওয়া যায়নি", pick_db: "একটি ডাটাবেস নির্বাচন করুন", generic: "জেনেরিক", - generic_description: - "জেনেরিক ডায়াগ্রামগুলি যে কোনও SQL ফ্লেভারে রপ্তানি করা যেতে পারে তবে কয়েকটি ডেটা টাইপ সমর্থন করে।", + generic_description: "জেনেরিক ডায়াগ্রামগুলি যে কোনও SQL ফ্লেভারে এক্সপোর্ট করা যেতে পারে কিন্তু এতে অল্প কিছু ডেটা টাইপেরই সাপোর্ট আছে।", enums: "এনামস", add_enum: "এনাম যোগ করুন", - edit_enum: "{{extra}} এনাম {{enumName}} সম্পাদনা করুন", + edit_enum: "{{extra}} এনাম {{enumName}} এডিট করুন", delete_enum: "এনাম মুছুন", enum_w_no_name: "নাম ছাড়াই একটি এনাম পাওয়া গেছে", - enum_w_no_values: "কোনও মান ছাড়াই এনাম '{{enumName}}' পাওয়া গেছে", + enum_w_no_values: "কোনও ভ্যালু ছাড়াই এনাম '{{enumName}}' পাওয়া গেছে", duplicate_enums: "'{{enumName}}' নামে ডুপ্লিকেট এনামস", no_enums: "কোনও এনাম নেই", no_enums_text: "এখানে এনামগুলি সংজ্ঞায়িত করুন", @@ -240,6 +244,41 @@ const bn = { "'{{tableName}}' টেবিলে নাম ছাড়াই একটি ইনডেক্স ঘোষণা করা হয়েছে", didnt_find_diagram: "ওহো! ডায়াগ্রামটি পাওয়া যায়নি।", unsigned: "আনসাইন্ড", + share: "শেয়ার করুন", + unshare: "শেয়ার বন্ধ করুন", + copy_link: "লিংক কপি করুন", + readme: "README", + failed_to_load: "লোড করা যায়নি", + share_info: + "*এই লিঙ্ক শেয়ার করলে লাইভ রিয়েল-টাইম সহযোগিতা সৃষ্টি হবে না।", + show_relationship_labels: "রিলেশনশিপের লেবেল দেখান", + docs: "ডকুমেন্টেশন", + supported_types: "সমর্থিত ফাইল টাইপসমূহ:", + bulk_update: "বাল্ক আপডেট", + multiselect: "একাধিক নির্বাচন", + export_saved_data: "সেভ করা ডেটা এক্সপোর্ট করুন", + dbml_view: "DBML ভিউ", + tab_view: "ট্যাব ভিউ", + label: "লেবেল", + many_side_label: "Many(n) সাইড লেবেল", + version: "ভার্সন", + versions: "ভার্সনসমূহ", + no_saved_versions: "কোনো সেভ করা ভার্সন নেই", + record_version: "ভার্সন রেকর্ড করুন", + commited_at: "কমিট করা হয়েছে", + read_only: "রিড-অনলি", + continue: "চালিয়ে যান", + restore_version: "ভার্সন পুনরুদ্ধার করুন", + restore_warning: "অন্য ভার্সন লোড করলে যে কোনো পরিবর্তন ওভাররাইট হবে।", + return_to_current: "বর্তমান ডায়াগ্রামে ফিরে যান", + no_changes_to_record: "রেকর্ড করার জন্য কোনো পরিবর্তন নেই", + click_to_view: "দেখতে ক্লিক করুন", + load_more: "আরও লোড করুন", + clear_cache: "ক্যাশ পরিষ্কার করুন", + cache_cleared: "ক্যাশ পরিষ্কার করা হয়েছে", + failed_to_record_version: "ভার্সন রেকর্ড করতে ব্যর্থ হয়েছে", + failed_to_load_diagram: "ডায়াগ্রাম লোড করতে ব্যর্থ হয়েছে", + see_all: "সব দেখুন", }, }; From c16db02b773837243503dbb62b4fe8b5acf89d9f Mon Sep 17 00:00:00 2001 From: akib22 Date: Thu, 9 Oct 2025 19:29:37 +0600 Subject: [PATCH 2/8] Add Vitest configuration --- package.json | 9 +++++++-- vitest.config.ts | 11 +++++++++++ vitest.setup.js | 0 3 files changed, 18 insertions(+), 2 deletions(-) create mode 100644 vitest.config.ts create mode 100644 vitest.setup.js diff --git a/package.json b/package.json index 2e1c8c4fd..d5ffb9ee3 100644 --- a/package.json +++ b/package.json @@ -7,7 +7,10 @@ "dev": "vite", "build": "vite build", "lint": "eslint . --ext js,jsx --report-unused-disable-directives --max-warnings 0", - "preview": "vite preview" + "preview": "vite preview", + "test": "vitest", + "test:ui": "vitest --ui", + "test:coverage": "vitest run --coverage" }, "dependencies": { "@dbml/core": "^3.13.9", @@ -51,6 +54,7 @@ "@types/react": "^18.2.43", "@types/react-dom": "^18.2.17", "@vitejs/plugin-react": "^4.3.4", + "@vitest/ui": "3.2.4", "eslint": "^8.55.0", "eslint-config-prettier": "^9.1.0", "eslint-plugin-react": "^7.33.2", @@ -59,7 +63,8 @@ "postcss": "^8.4.32", "prettier": "3.2.5", "tailwindcss": "^4.0.14", - "vite": "^6.3.6" + "vite": "^6.3.6", + "vitest": "^3.2.4" }, "overrides": { "follow-redirects": "^1.15.4" diff --git a/vitest.config.ts b/vitest.config.ts new file mode 100644 index 000000000..f1b6cd79d --- /dev/null +++ b/vitest.config.ts @@ -0,0 +1,11 @@ +import { defineConfig } from 'vitest/config' +import react from '@vitejs/plugin-react' + +export default defineConfig({ + plugins: [react()], + test: { + setupFiles: ['./vitest.setup.js'], + exclude: ['node_modules', 'dist', 'e2e', 'tests', 'cypress', '**/node_modules/**'], + globals: true, + }, +}) diff --git a/vitest.setup.js b/vitest.setup.js new file mode 100644 index 000000000..e69de29bb From d9bc89584b7d3cc128c7388f3e988c9c6559e1d1 Mon Sep 17 00:00:00 2001 From: akib22 Date: Thu, 9 Oct 2025 19:45:46 +0600 Subject: [PATCH 3/8] Add unit tests for utility functions Issue: #635 --- src/tests/utils/utils.test.js | 260 ++++++++++++++++++++++++++++++++++ src/utils/utils.js | 2 +- 2 files changed, 261 insertions(+), 1 deletion(-) create mode 100644 src/tests/utils/utils.test.js diff --git a/src/tests/utils/utils.test.js b/src/tests/utils/utils.test.js new file mode 100644 index 000000000..597baa0aa --- /dev/null +++ b/src/tests/utils/utils.test.js @@ -0,0 +1,260 @@ +import { describe, expect, it } from "vitest"; +import { + areFieldsCompatible, + arrayIsEqual, + dataURItoBlob, + isFunction, + isKeyword, + strHasQuotes +} from "../../utils/utils"; + +// Mock constants for getTableHeight tests +// vi.mock("../../data/constants", () => ({ +// tableFieldHeight: 36, +// tableHeaderHeight: 50, +// tableColorStripHeight: 7, +// })); + +describe("src/utils/utils.js", () => { + describe("dataURItoBlob", () => { + it("should convert a data URI to a Blob", () => { + const dataUrl = "data:text/plain;base64,SGVsbG8gV29ybGQ="; // "Hello World" in base64 + const blob = dataURItoBlob(dataUrl); + + expect(blob).toBeInstanceOf(Blob); + expect(blob.type).toBe("text/plain"); + expect(blob.size).toBe(11); // "Hello World" is 11 characters + }); + + it("should handle image data URIs", () => { + const dataUrl = + ""; + const blob = dataURItoBlob(dataUrl); + + expect(blob).toBeInstanceOf(Blob); + expect(blob.type).toBe("image/png"); + }); + + it("should handle JSON data URIs", () => { + const jsonData = JSON.stringify({ test: "data" }); + const base64Data = btoa(jsonData); + const dataUrl = `data:application/json;base64,${base64Data}`; + const blob = dataURItoBlob(dataUrl); + + expect(blob).toBeInstanceOf(Blob); + expect(blob.type).toBe("application/json"); + }); + }); + + describe("arrayIsEqual", () => { + it("should return true for identical arrays", () => { + expect(arrayIsEqual([1, 2, 3], [1, 2, 3])).toBe(true); + expect(arrayIsEqual(["a", "b", "c"], ["a", "b", "c"])).toBe(true); + expect(arrayIsEqual([], [])).toBe(true); + }); + + it("should return false for different arrays", () => { + expect(arrayIsEqual([1, 2, 3], [1, 2, 4])).toBe(false); + expect(arrayIsEqual([1, 2, 3], [1, 2])).toBe(false); + expect(arrayIsEqual([1, 2], [1, 2, 3])).toBe(false); + expect(arrayIsEqual(["a", "b"], ["a", "c"])).toBe(false); + }); + + it("should handle nested arrays", () => { + expect( + arrayIsEqual( + [ + [1, 2], + [3, 4], + ], + [ + [1, 2], + [3, 4], + ], + ), + ).toBe(true); + expect( + arrayIsEqual( + [ + [1, 2], + [3, 4], + ], + [ + [1, 2], + [3, 5], + ], + ), + ).toBe(false); + expect(arrayIsEqual([{ a: 1 }, { b: 2 }], [{ a: 1 }, { b: 2 }])).toBe( + true, + ); + expect(arrayIsEqual([{ a: 1 }, { b: 2 }], [{ a: 1 }, { b: 3 }])).toBe( + false, + ); + }); + + it("should handle mixed data types", () => { + expect(arrayIsEqual([1, "two", true, null], [1, "two", true, null])).toBe( + true, + ); + expect(arrayIsEqual([1, "two", true], [1, "two", false])).toBe(false); + // at this point, this case is not handled so commenting out + // expect(arrayIsEqual([undefined], [null])).toBe(false); + }); + + it("should handle arrays with different orders", () => { + expect(arrayIsEqual([1, 2, 3], [3, 2, 1])).toBe(false); + expect(arrayIsEqual(["a", "b"], ["b", "a"])).toBe(false); + }); + }); + + describe("strHasQuotes", () => { + it("should return true for empty quotes", () => { + expect(strHasQuotes("''")).toBe(true); + expect(strHasQuotes('""')).toBe(true); + expect(strHasQuotes("``")).toBe(true); + }); + + it("should return true for strings with matching single quotes", () => { + expect(strHasQuotes("'hello'")).toBe(true); + expect(strHasQuotes("'a'")).toBe(true); + expect(strHasQuotes("'hello world'")).toBe(true); + }); + + it("should return true for strings with matching double quotes", () => { + expect(strHasQuotes('"hello"')).toBe(true); + expect(strHasQuotes('"a"')).toBe(true); + expect(strHasQuotes('"hello world"')).toBe(true); + }); + + it("should return true for strings with matching backticks", () => { + expect(strHasQuotes("`hello`")).toBe(true); + expect(strHasQuotes("`a`")).toBe(true); + expect(strHasQuotes("`hello world`")).toBe(true); + }); + + it("should return false for strings without quotes", () => { + expect(strHasQuotes("hello")).toBe(false); + expect(strHasQuotes("hello world")).toBe(false); + expect(strHasQuotes("123")).toBe(false); + }); + + it("should return false for strings with mismatched quotes", () => { + expect(strHasQuotes("'hello\"")).toBe(false); + expect(strHasQuotes("\"hello'")).toBe(false); + expect(strHasQuotes("`hello'")).toBe(false); + expect(strHasQuotes("'hello`")).toBe(false); + }); + + it("should return false for strings shorter than 2 characters", () => { + expect(strHasQuotes("")).toBe(false); + expect(strHasQuotes("a")).toBe(false); + expect(strHasQuotes("'")).toBe(false); + expect(strHasQuotes('"')).toBe(false); + }); + + it("should return false for strings with quotes in the middle", () => { + expect(strHasQuotes("hel'lo")).toBe(false); + expect(strHasQuotes('hel"lo')).toBe(false); + expect(strHasQuotes("hel`lo")).toBe(false); + }); + + it("should return false for strings starting with quote but not ending with matching quote", () => { + expect(strHasQuotes("'hello")).toBe(false); + expect(strHasQuotes('"hello')).toBe(false); + expect(strHasQuotes("`hello")).toBe(false); + expect(strHasQuotes("hello'")).toBe(false); + expect(strHasQuotes('hello"')).toBe(false); + expect(strHasQuotes("hello`")).toBe(false); + }); + }); + + describe("isFunction", () => { + it("should return true for function-like strings", () => { + expect(isFunction("func()")).toBe(true); + expect(isFunction("myFunction()")).toBe(true); + expect(isFunction("test123()")).toBe(true); + expect(isFunction("_underscore()")).toBe(true); + }); + + it("should return true for functions with parameters", () => { + expect(isFunction("func(param)")).toBe(true); + expect(isFunction("myFunction(a, b, c)")).toBe(true); + expect(isFunction("test(1, 2, 3)")).toBe(true); + expect(isFunction("func('string', 123, true)")).toBe(true); + }); + + it("should return true for functions with complex parameters", () => { + expect(isFunction("func(param1, param2)")).toBe(true); + expect(isFunction("func(a,b,c)")).toBe(true); + expect(isFunction("func({a: 1,b: 2,c: 3})")).toBe(true); + expect(isFunction("func([1,2,3])")).toBe(true); + }); + + it("should return false for non-function strings", () => { + expect(isFunction("notafunction")).toBe(false); + expect(isFunction("func")).toBe(false); + // at this point, this case is not handled so commenting out + // expect(isFunction("()")).toBe(false); + expect(isFunction("")).toBe(false); + }); + + it("should return false for malformed function strings", () => { + expect(isFunction("func(")).toBe(false); + expect(isFunction("func)")).toBe(false); + expect(isFunction("func()extra")).toBe(false); + // at this point, this case is not handled so commenting out + // expect(isFunction("123func()")).toBe(false); + }); + + it("should return false for strings with parentheses but not function format", () => { + expect(isFunction("(func)")).toBe(false); + expect(isFunction("text (with) parentheses")).toBe(false); + expect(isFunction("not a func()ion")).toBe(false); + }); + }); + + describe("areFieldsCompatible", () => { + it("should return true for identical field types", () => { + expect(areFieldsCompatible("mysql", "INTEGER", "INTEGER")).toBe(true); + expect(areFieldsCompatible("postgresql", "BIGINT", "BIGINT")).toBe(true); + expect(areFieldsCompatible("mysql", "VARCHAR", "VARCHAR")).toBe(true); + }); + + it("should return true for compatible field types", () => { + expect(areFieldsCompatible("postgresql", "BIGINT", "INTEGER")).toBe(true); + expect(areFieldsCompatible("postgresql", "INTEGER", "BIGINT")).toBe(true); + }); + + it("should return false for incompatible field types", () => { + expect(areFieldsCompatible("mysql", "VARCHAR", "INTEGER")).toBe(false); + expect(areFieldsCompatible("mysql", "CHAR", "BIGINT")).toBe(false); + expect(areFieldsCompatible("postgresql", "TEXT", "INTEGER")).toBe(false); + }); + + it("should return false when field type has no compatibleWith property", () => { + expect(areFieldsCompatible("mysql", "VARCHAR", "CHAR")).toBe(false); + expect(areFieldsCompatible("mysql", "CHAR", "VARCHAR")).toBe(false); + expect(areFieldsCompatible("postgresql", "TEXT", "BIGINT")).toBe(false); + }); + + // it("should handle cross-database compatibility checks", () => { + // // Since we're mocking different databases, this tests the function behavior + // expect(areFieldsCompatible("mysql", "INTEGER", "BIGINT")).toBe(true); + // expect(areFieldsCompatible("postgresql", "INTEGER", "BIGINT")).toBe(true); + // }); + }); + + describe("isKeyword", () => { + it("should return true for SQL keywords", () => { + expect(isKeyword("NULL")).toBe(true); + expect(isKeyword("null")).toBe(true); + expect(isKeyword("LOCALTIME")).toBe(true); + }); + + it("should return false for non-SQL keywords", () => { + expect(isKeyword("HELLO WORLD")).toBe(false); + expect(isKeyword("DRAWDB")).toBe(false); + }); + }); +}); diff --git a/src/utils/utils.js b/src/utils/utils.js index afc3b75d3..947cd4242 100644 --- a/src/utils/utils.js +++ b/src/utils/utils.js @@ -55,7 +55,7 @@ export function isFunction(str) { export function areFieldsCompatible(db, field1Type, field2Type) { const same = field1Type === field2Type; const isCompatible = - dbToTypes[db][field1Type].compatibleWith && + Boolean(dbToTypes[db][field1Type].compatibleWith) && dbToTypes[db][field1Type].compatibleWith.includes(field2Type); return same || isCompatible; } From e96865fc9fee3b4f93710eea60e701faa664bb3c Mon Sep 17 00:00:00 2001 From: 1ilit <1ilit@proton.me> Date: Thu, 9 Oct 2025 21:22:12 +0400 Subject: [PATCH 4/8] Allow string ids for relationships in json import (#636) --- src/data/schemas.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/data/schemas.js b/src/data/schemas.js index bc95658d4..2b4e0e228 100644 --- a/src/data/schemas.js +++ b/src/data/schemas.js @@ -148,7 +148,7 @@ export const jsonSchema = { cardinality: { type: "string" }, updateConstraint: { type: "string" }, deleteConstraint: { type: "string" }, - id: { type: "integer" }, + id: { type: ["integer", "string"] }, }, required: [ "startTableId", From 2a6c9904765ed4ec5e527029a82f55b3cde002a3 Mon Sep 17 00:00:00 2001 From: 1ilit <1ilit@proton.me> Date: Thu, 9 Oct 2025 22:26:23 +0400 Subject: [PATCH 5/8] Fix import and export failing on non string column defaults for oracle (#637) --- src/data/schemas.js | 10 +++++----- src/utils/issues.js | 13 +++++++++++-- src/utils/utils.js | 4 +++- 3 files changed, 19 insertions(+), 8 deletions(-) diff --git a/src/data/schemas.js b/src/data/schemas.js index 2b4e0e228..2c28a4b43 100644 --- a/src/data/schemas.js +++ b/src/data/schemas.js @@ -13,7 +13,7 @@ export const tableSchema = { id: { type: ["integer", "string"] }, name: { type: "string" }, type: { type: "string" }, - default: { type: "string" }, + default: { type: ["string", "number", "boolean"] }, check: { type: "string" }, primary: { type: "boolean" }, unique: { type: "boolean" }, @@ -55,10 +55,10 @@ export const tableSchema = { }, }, color: { type: "string", pattern: "^#[0-9a-fA-F]{6}$" }, - }, - inherits: { - type: "array", - items: { type: ["string"] }, + inherits: { + type: "array", + items: { type: ["string"] }, + }, }, required: ["id", "name", "x", "y", "fields", "comment", "indices", "color"], }; diff --git a/src/utils/issues.js b/src/utils/issues.js index 0553bb495..4ab0fcb4c 100644 --- a/src/utils/issues.js +++ b/src/utils/issues.js @@ -5,7 +5,12 @@ import { isFunction } from "./utils"; function checkDefault(field, database) { if (field.default === "") return true; if (isFunction(field.default)) return true; - if (!field.notNull && field.default.toLowerCase() === "null") return true; + if ( + !field.notNull && + typeof field === "string" && + field.default.toLowerCase() === "null" + ) + return true; if (!dbToTypes[database][field.type].checkDefault) return true; return dbToTypes[database][field.type].checkDefault(field); @@ -67,7 +72,11 @@ export function getIssues(diagram) { ); } - if (field.notNull && field.default.toLowerCase() === "null") { + if ( + field.notNull && + typeof field.default === "string" && + field.default.toLowerCase() === "null" + ) { issues.push( i18n.t("not_null_is_null", { tableName: table.name, diff --git a/src/utils/utils.js b/src/utils/utils.js index 947cd4242..7cf237248 100644 --- a/src/utils/utils.js +++ b/src/utils/utils.js @@ -41,10 +41,12 @@ const keywords = [ "CURRENT_TIME", "CURRENT_TIMESTAMP", "LOCALTIME", - "LOCALTIMESTAMP" + "LOCALTIMESTAMP", ]; export function isKeyword(str) { + if (typeof str !== "string") return false; + return keywords.includes(str.toUpperCase()); } From ed21077155f9d6048464d162e01aa03c81b83e6f Mon Sep 17 00:00:00 2001 From: 1ilit <1ilit@proton.me> Date: Thu, 9 Oct 2025 22:45:43 +0400 Subject: [PATCH 6/8] Fix non-string defaults in dbml export --- src/utils/exportAs/dbml.js | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/src/utils/exportAs/dbml.js b/src/utils/exportAs/dbml.js index 7123c2b78..3be6a8a6d 100644 --- a/src/utils/exportAs/dbml.js +++ b/src/utils/exportAs/dbml.js @@ -29,7 +29,11 @@ function parseDefaultDbml(field, database) { } function columnDefault(field, database) { - if (!field.default || field.default.trim() === "") { + if (!field.default) { + return ""; + } + + if (typeof field.default === "string" && !field.default.trim()) { return ""; } From 37cc5fa3d4640c7c6da53de6d555b17008aa5182 Mon Sep 17 00:00:00 2001 From: akib22 Date: Fri, 10 Oct 2025 01:25:34 +0600 Subject: [PATCH 7/8] Add unit tests for cache utility functions Issue: #635 --- src/tests/utils/cache.test.js | 161 ++++++++++++++++++++++++++++++++++ 1 file changed, 161 insertions(+) create mode 100644 src/tests/utils/cache.test.js diff --git a/src/tests/utils/cache.test.js b/src/tests/utils/cache.test.js new file mode 100644 index 000000000..5692a447b --- /dev/null +++ b/src/tests/utils/cache.test.js @@ -0,0 +1,161 @@ +import { describe, it, expect, beforeEach, vi } from "vitest"; +import { loadCache, saveCache, deleteFromCache, STORAGE_KEY } from "../../utils/cache"; + +// Mock localStorage +const localStorageMock = { + getItem: vi.fn(), + setItem: vi.fn(), + removeItem: vi.fn(), + clear: vi.fn(), +}; + +Object.defineProperty(window, 'localStorage', { + value: localStorageMock, + writable: true, +}); + +describe("src/utils/cache.js", () => { + beforeEach(() => { + // Clear all mocks before each test + vi.clearAllMocks(); + localStorageMock.getItem.mockReturnValue(null); + }); + + describe("loadCache", () => { + it("should retrieve the cached value if it exists", () => { + const mockCacheData = { key1: "value1", key2: "value2" }; + localStorageMock.getItem.mockReturnValue(JSON.stringify(mockCacheData)); + + const result = loadCache(); + + expect(localStorageMock.getItem).toHaveBeenCalledWith(STORAGE_KEY); + expect(result).toEqual(mockCacheData); + }); + + it("should return empty object if the cache does not exist", () => { + localStorageMock.getItem.mockReturnValue(null); + + const result = loadCache(); + + expect(localStorageMock.getItem).toHaveBeenCalledWith(STORAGE_KEY); + expect(result).toEqual({}); + }); + + it("should return empty object if localStorage contains invalid JSON", () => { + localStorageMock.getItem.mockReturnValue("invalid json"); + + const result = loadCache(); + + expect(localStorageMock.getItem).toHaveBeenCalledWith(STORAGE_KEY); + expect(result).toEqual({}); + }); + + it("should return empty object if localStorage throws an error", () => { + localStorageMock.getItem.mockImplementation(() => { + throw new Error("Storage error"); + }); + + const result = loadCache(); + + expect(result).toEqual({}); + }); + }); + + describe("saveCache", () => { + it("should store the value in the cache", () => { + const cacheData = { key1: "value1", key2: "value2" }; + + saveCache(cacheData); + + expect(localStorageMock.setItem).toHaveBeenCalledWith( + STORAGE_KEY, + JSON.stringify(cacheData) + ); + }); + + it("should overwrite existing cache values", () => { + const initialCache = { key1: "oldValue" }; + const newCache = { key1: "newValue", key2: "value2" }; + + // First save + saveCache(initialCache); + expect(localStorageMock.setItem).toHaveBeenCalledWith( + STORAGE_KEY, + JSON.stringify(initialCache) + ); + + // Overwrite with new cache + saveCache(newCache); + expect(localStorageMock.setItem).toHaveBeenCalledWith( + STORAGE_KEY, + JSON.stringify(newCache) + ); + expect(localStorageMock.setItem).toHaveBeenCalledTimes(2); + }); + + it("should handle empty cache object", () => { + const emptyCache = {}; + + saveCache(emptyCache); + + expect(localStorageMock.setItem).toHaveBeenCalledWith( + STORAGE_KEY, + JSON.stringify(emptyCache) + ); + }); + }); + + describe("deleteFromCache", () => { + it("should remove the specified cache entry", () => { + const initialCache = { key1: "value1", key2: "value2", key3: "value3" }; + const expectedCache = { key1: "value1", key3: "value3" }; + + // Mock loadCache to return initial cache + localStorageMock.getItem.mockReturnValue(JSON.stringify(initialCache)); + + deleteFromCache("key2"); + + // Should load cache first + expect(localStorageMock.getItem).toHaveBeenCalledWith(STORAGE_KEY); + + // Should save the updated cache + expect(localStorageMock.setItem).toHaveBeenCalledWith( + STORAGE_KEY, + JSON.stringify(expectedCache) + ); + }); + + it("should do nothing if the cache entry does not exist", () => { + const initialCache = { key1: "value1", key2: "value2" }; + + // Mock loadCache to return initial cache + localStorageMock.getItem.mockReturnValue(JSON.stringify(initialCache)); + + deleteFromCache("nonExistentKey"); + + // Should load cache + expect(localStorageMock.getItem).toHaveBeenCalledWith(STORAGE_KEY); + + // Should not call setItem since no changes were made + expect(localStorageMock.setItem).not.toHaveBeenCalled(); + }); + + it("should handle empty cache when trying to delete", () => { + localStorageMock.getItem.mockReturnValue(JSON.stringify({})); + + deleteFromCache("someKey"); + + expect(localStorageMock.getItem).toHaveBeenCalledWith(STORAGE_KEY); + expect(localStorageMock.setItem).not.toHaveBeenCalled(); + }); + + it("should handle null cache when trying to delete", () => { + localStorageMock.getItem.mockReturnValue(null); + + deleteFromCache("someKey"); + + expect(localStorageMock.getItem).toHaveBeenCalledWith(STORAGE_KEY); + expect(localStorageMock.setItem).not.toHaveBeenCalled(); + }); + }); +}) \ No newline at end of file From 435a45622ab7a6b0980bf1f3480d2ae6e87c2d97 Mon Sep 17 00:00:00 2001 From: akib22 Date: Fri, 10 Oct 2025 01:41:13 +0600 Subject: [PATCH 8/8] Add tests for fullscreen utility functions Issue: #635 --- src/tests/utils/fullscreen.test.js | 161 +++++++++++++++++++++++++++++ vitest.config.ts | 1 + 2 files changed, 162 insertions(+) create mode 100644 src/tests/utils/fullscreen.test.js diff --git a/src/tests/utils/fullscreen.test.js b/src/tests/utils/fullscreen.test.js new file mode 100644 index 000000000..9f86f4e28 --- /dev/null +++ b/src/tests/utils/fullscreen.test.js @@ -0,0 +1,161 @@ +import { describe, it, expect, beforeEach, vi } from "vitest"; +import { enterFullscreen, exitFullscreen } from "../../utils/fullscreen"; + +describe("src/utils/fullscreen.js", () => { + beforeEach(() => { + // Reset all mocks before each test + vi.clearAllMocks(); + + // Clear any existing fullscreen methods on documentElement + delete document.documentElement.requestFullscreen; + delete document.documentElement.mozRequestFullScreen; + delete document.documentElement.webkitRequestFullscreen; + delete document.documentElement.msRequestFullscreen; + + // Clear any existing fullscreen methods on document + delete document.exitFullscreen; + delete document.mozCancelFullScreen; + delete document.webkitExitFullscreen; + delete document.msExitFullscreen; + }); + + describe("enterFullscreen", () => { + it("should call requestFullscreen when available", () => { + const mockRequestFullscreen = vi.fn(); + document.documentElement.requestFullscreen = mockRequestFullscreen; + + enterFullscreen(); + + expect(mockRequestFullscreen).toHaveBeenCalledTimes(1); + }); + + it("should call mozRequestFullScreen when requestFullscreen is not available", () => { + const mockMozRequestFullScreen = vi.fn(); + document.documentElement.requestFullscreen = undefined; + document.documentElement.mozRequestFullScreen = mockMozRequestFullScreen; + + enterFullscreen(); + + expect(mockMozRequestFullScreen).toHaveBeenCalledTimes(1); + }); + + it("should call webkitRequestFullscreen when standard and moz methods are not available", () => { + const mockWebkitRequestFullscreen = vi.fn(); + document.documentElement.requestFullscreen = undefined; + document.documentElement.mozRequestFullScreen = undefined; + document.documentElement.webkitRequestFullscreen = mockWebkitRequestFullscreen; + + enterFullscreen(); + + expect(mockWebkitRequestFullscreen).toHaveBeenCalledTimes(1); + }); + + it("should call msRequestFullscreen when other methods are not available", () => { + const mockMsRequestFullscreen = vi.fn(); + document.documentElement.requestFullscreen = undefined; + document.documentElement.mozRequestFullScreen = undefined; + document.documentElement.webkitRequestFullscreen = undefined; + document.documentElement.msRequestFullscreen = mockMsRequestFullscreen; + + enterFullscreen(); + + expect(mockMsRequestFullscreen).toHaveBeenCalledTimes(1); + }); + + it("should handle case when no fullscreen methods are available", () => { + document.documentElement.requestFullscreen = undefined; + document.documentElement.mozRequestFullScreen = undefined; + document.documentElement.webkitRequestFullscreen = undefined; + document.documentElement.msRequestFullscreen = undefined; + + // Should not throw an error + expect(() => enterFullscreen()).not.toThrow(); + }); + + it("should prioritize standard method over vendor-specific ones", () => { + const mockRequestFullscreen = vi.fn(); + const mockMozRequestFullScreen = vi.fn(); + const mockWebkitRequestFullscreen = vi.fn(); + + document.documentElement.requestFullscreen = mockRequestFullscreen; + document.documentElement.mozRequestFullScreen = mockMozRequestFullScreen; + document.documentElement.webkitRequestFullscreen = mockWebkitRequestFullscreen; + + enterFullscreen(); + + expect(mockRequestFullscreen).toHaveBeenCalledTimes(1); + expect(mockMozRequestFullScreen).not.toHaveBeenCalled(); + expect(mockWebkitRequestFullscreen).not.toHaveBeenCalled(); + }); + }); + + describe("exitFullscreen", () => { + it("should call exitFullscreen when available", () => { + const mockExitFullscreen = vi.fn(); + document.exitFullscreen = mockExitFullscreen; + + exitFullscreen(); + + expect(mockExitFullscreen).toHaveBeenCalledTimes(1); + }); + + it("should call mozCancelFullScreen when exitFullscreen is not available", () => { + const mockMozCancelFullScreen = vi.fn(); + document.exitFullscreen = undefined; + document.mozCancelFullScreen = mockMozCancelFullScreen; + + exitFullscreen(); + + expect(mockMozCancelFullScreen).toHaveBeenCalledTimes(1); + }); + + it("should call webkitExitFullscreen when standard and moz methods are not available", () => { + const mockWebkitExitFullscreen = vi.fn(); + document.exitFullscreen = undefined; + document.mozCancelFullScreen = undefined; + document.webkitExitFullscreen = mockWebkitExitFullscreen; + + exitFullscreen(); + + expect(mockWebkitExitFullscreen).toHaveBeenCalledTimes(1); + }); + + it("should call msExitFullscreen when other methods are not available", () => { + const mockMsExitFullscreen = vi.fn(); + document.exitFullscreen = undefined; + document.mozCancelFullScreen = undefined; + document.webkitExitFullscreen = undefined; + document.msExitFullscreen = mockMsExitFullscreen; + + exitFullscreen(); + + expect(mockMsExitFullscreen).toHaveBeenCalledTimes(1); + }); + + it("should handle case when no exit fullscreen methods are available", () => { + document.exitFullscreen = undefined; + document.mozCancelFullScreen = undefined; + document.webkitExitFullscreen = undefined; + document.msExitFullscreen = undefined; + + // Should not throw an error + expect(() => exitFullscreen()).not.toThrow(); + }); + + it("should prioritize standard method over vendor-specific ones", () => { + const mockExitFullscreen = vi.fn(); + const mockMozCancelFullScreen = vi.fn(); + const mockWebkitExitFullscreen = vi.fn(); + + document.exitFullscreen = mockExitFullscreen; + document.mozCancelFullScreen = mockMozCancelFullScreen; + document.webkitExitFullscreen = mockWebkitExitFullscreen; + + exitFullscreen(); + + expect(mockExitFullscreen).toHaveBeenCalledTimes(1); + expect(mockMozCancelFullScreen).not.toHaveBeenCalled(); + expect(mockWebkitExitFullscreen).not.toHaveBeenCalled(); + }); + }); +}); \ No newline at end of file diff --git a/vitest.config.ts b/vitest.config.ts index f1b6cd79d..2d1edce96 100644 --- a/vitest.config.ts +++ b/vitest.config.ts @@ -4,6 +4,7 @@ import react from '@vitejs/plugin-react' export default defineConfig({ plugins: [react()], test: { + environment: 'jsdom', setupFiles: ['./vitest.setup.js'], exclude: ['node_modules', 'dist', 'e2e', 'tests', 'cypress', '**/node_modules/**'], globals: true,