diff --git a/.cursor/rules/bulma-usage.mdc b/.cursor/rules/bulma-usage.mdc new file mode 100644 index 0000000..6547834 --- /dev/null +++ b/.cursor/rules/bulma-usage.mdc @@ -0,0 +1,107 @@ +--- +description: Standards for using Bulma CSS framework consistently across the application +globs: **/*.{svelte,css,html} +--- + +When styling components: + +1. Use Bulma's Built-in Classes: + - Prefer Bulma utility classes over custom CSS + - Use spacing utilities (m*, p*) for margins and padding + - Use color utilities for consistent theming + - Use responsive helpers for different screen sizes + - Use flexbox utilities for layout + +2. Component Structure: + - Follow Bulma's component structure exactly + - Use proper nesting (e.g., navbar > navbar-brand > navbar-item) + - Keep modifier classes consistent with Bulma's patterns + - Use semantic Bulma classes (e.g., 'button is-primary' not custom colors) + +3. Custom Styles: + - Only add custom CSS when Bulma doesn't provide the functionality + - Document why custom CSS is needed + - Use Bulma's CSS variables for theming + - Keep custom styles minimal and focused + +4. Dark Mode: + - Use Bulma's CSS variables for theming + - Override colors using CSS variables, not direct colors + - Maintain consistent contrast ratios + - Test dark mode for all components + +5. Responsive Design: + - Use Bulma's responsive classes (is-*-mobile, is-*-tablet, etc.) + - Test all breakpoints + - Maintain consistent spacing across screen sizes + - Use proper container classes + +6. Common Patterns: + - Forms: field > label + control > input + - Buttons: button + is-* modifiers + - Cards: card > card-header + card-content + card-footer + - Navigation: navbar with proper structure + - Grid: columns > column with proper sizes + +7. Performance: + - Don't duplicate Bulma styles + - Minimize custom CSS + - Use Bulma's minified version + - Remove unused Bulma features if size is a concern + +metadata: + priority: high + version: 1.0 + + +examples: + - input: | + # Bad: Custom styles for spacing + + output: | + +
...
+ + - input: | + # Bad: Custom color styles + + output: | + + + + - input: | + # Bad: Custom responsive styles + + output: | + +
...
+ + - input: | + # Bad: Inconsistent form structure +
+ + +
+ output: | + +
+ +
+ +
+
+ \ No newline at end of file diff --git a/.cursor/rules/frontend-testing.mdc b/.cursor/rules/frontend-testing.mdc new file mode 100644 index 0000000..db7b86e --- /dev/null +++ b/.cursor/rules/frontend-testing.mdc @@ -0,0 +1,80 @@ +--- +description: Frontend testing standards using Playwright for end-to-end testing +globs: **/*.{ts,js,svelte} +--- + +When working on frontend features: + +1. Test Coverage Requirements: + - All new features must have e2e tests + - Critical user flows must be tested + - Test both success and error paths + - Test responsive behavior for mobile/desktop + - Test accessibility where applicable + +2. Test Structure: + - Use descriptive test names + - Group related tests in describe blocks + - Use beforeEach for common setup + - Clean up test data after each test + - Keep tests focused and atomic + +3. Testing Best Practices: + - Test user interactions, not implementation + - Use data-testid for test-specific selectors + - Test across different viewport sizes + - Test across multiple browsers + - Verify visual and functional aspects + +4. Required Test Cases: + - Navigation flows + - Form submissions + - Error handling + - Loading states + - Responsive behavior + - Cross-browser compatibility + +5. Running Tests: + - Run tests before committing: `bun test` + - Debug tests with UI: `bun test:ui` + - Debug specific test: `bun test:debug` + - Install browsers: `bun test:install` + +6. CI Integration: + - Tests must pass in CI + - Screenshots on failure + - HTML test reports + - Retry failed tests + - Parallel execution where possible + +metadata: + priority: high + version: 1.0 + + +examples: + - input: | + # Bad: No tests for new feature + git commit -m "feat: add note search" + output: | + # Good: Feature with tests + bun test + git commit -m "feat: add note search with e2e tests + + Tests added: + - Search functionality + - Empty results handling + - Error states + - Mobile layout" + + - input: | + # Bad: Testing implementation details + test('internal state is correct', () => { + expect(component.state.value).toBe(true) + }) + output: | + # Good: Testing user interaction + test('user can toggle feature', async ({ page }) => { + await page.click('[data-testid=toggle]') + await expect(page.locator('.status')).toHaveText('Enabled') + }) \ No newline at end of file diff --git a/frontend/bun.lock b/frontend/bun.lock index 2f90c69..d296bfc 100644 --- a/frontend/bun.lock +++ b/frontend/bun.lock @@ -11,6 +11,7 @@ "devDependencies": { "@eslint/compat": "^1.2.5", "@eslint/js": "^9.18.0", + "@playwright/test": "^1.40.0", "@sveltejs/adapter-static": "^3.0.8", "@sveltejs/kit": "^2.5.27", "@sveltejs/vite-plugin-svelte": "^5.0.0", @@ -124,6 +125,8 @@ "@nodelib/fs.walk": ["@nodelib/fs.walk@1.2.8", "", { "dependencies": { "@nodelib/fs.scandir": "2.1.5", "fastq": "^1.6.0" } }, "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg=="], + "@playwright/test": ["@playwright/test@1.50.1", "", { "dependencies": { "playwright": "1.50.1" }, "bin": { "playwright": "cli.js" } }, "sha512-Jii3aBg+CEDpgnuDxEp/h7BimHcUTDlpEtce89xEumlJ5ef2hqepZ+PWp1DDpYC/VO9fmWVI1IlEaoI5fK9FXQ=="], + "@polka/url": ["@polka/url@1.0.0-next.28", "", {}, "sha512-8LduaNlMZGwdZ6qWrKlfa+2M4gahzFkprZiAt2TF8uS0qQgBizKXpXURqvTJ4WtmupWxaLqjRb2UCTe72mu+Aw=="], "@rollup/rollup-android-arm-eabi": ["@rollup/rollup-android-arm-eabi@4.34.8", "", { "os": "android", "cpu": "arm" }, "sha512-q217OSE8DTp8AFHuNHXo0Y86e1wtlfVrXiAlwkIvGRQv9zbc6mE3sjIVfwI8sYUyNxwOg0j/Vm1RKM04JcWLJw=="], @@ -390,6 +393,10 @@ "picomatch": ["picomatch@2.3.1", "", {}, "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA=="], + "playwright": ["playwright@1.50.1", "", { "dependencies": { "playwright-core": "1.50.1" }, "optionalDependencies": { "fsevents": "2.3.2" }, "bin": { "playwright": "cli.js" } }, "sha512-G8rwsOQJ63XG6BbKj2w5rHeavFjy5zynBA9zsJMMtBoe/Uf757oG12NXz6e6OirF7RCrTVAKFXbLmn1RbL7Qaw=="], + + "playwright-core": ["playwright-core@1.50.1", "", { "bin": { "playwright-core": "cli.js" } }, "sha512-ra9fsNWayuYumt+NiM069M6OkcRb1FZSK8bgi66AtpFoWkg2+y0bJSNmkFrWhMbEBbVKC/EruAHH3g0zmtwGmQ=="], + "postcss": ["postcss@8.5.2", "", { "dependencies": { "nanoid": "^3.3.8", "picocolors": "^1.1.1", "source-map-js": "^1.2.1" } }, "sha512-MjOadfU3Ys9KYoX0AdkBlFEF1Vx37uCCeN4ZHnmwm9FfpbsGWMZeBLMmmpY+6Ocqod7mkdZ0DT31OlbsFrLlkA=="], "postcss-load-config": ["postcss-load-config@3.1.4", "", { "dependencies": { "lilconfig": "^2.0.5", "yaml": "^1.10.2" }, "peerDependencies": { "postcss": ">=8.0.9", "ts-node": ">=9.0.0" }, "optionalPeers": ["postcss", "ts-node"] }, "sha512-6DiM4E7v4coTE4uzA8U//WhtPwyhiim3eyjEMFCnUpzbrkK9wJHgKDT2mR+HbtSrd/NubVaYTOpSpjUl8NQeRg=="], @@ -488,6 +495,8 @@ "fast-glob/glob-parent": ["glob-parent@5.1.2", "", { "dependencies": { "is-glob": "^4.0.1" } }, "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow=="], + "playwright/fsevents": ["fsevents@2.3.2", "", { "os": "darwin" }, "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA=="], + "svelte-eslint-parser/eslint-scope": ["eslint-scope@7.2.2", "", { "dependencies": { "esrecurse": "^4.3.0", "estraverse": "^5.2.0" } }, "sha512-dOt21O7lTMhDM+X9mB4GX+DZrZtCUJPL/wlcTqxyrx5IvO0IYtILdtrQGQp+8n5S0gwSVmOf9NQrjMOgfQZlIg=="], "svelte-eslint-parser/eslint-visitor-keys": ["eslint-visitor-keys@3.4.3", "", {}, "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag=="], diff --git a/frontend/package.json b/frontend/package.json index 7de692c..0e64dda 100644 --- a/frontend/package.json +++ b/frontend/package.json @@ -11,11 +11,16 @@ "check": "svelte-kit sync && svelte-check --tsconfig ./tsconfig.json", "check:watch": "svelte-kit sync && svelte-check --tsconfig ./tsconfig.json --watch", "format": "prettier --write .", - "lint": "prettier --check . && eslint ." + "lint": "prettier --check . && eslint .", + "test": "playwright test", + "test:ui": "playwright test --ui", + "test:debug": "playwright test --debug", + "test:install": "playwright install" }, "devDependencies": { "@eslint/compat": "^1.2.5", "@eslint/js": "^9.18.0", + "@playwright/test": "1.50.1", "@sveltejs/adapter-static": "^3.0.8", "@sveltejs/kit": "^2.5.27", "@sveltejs/vite-plugin-svelte": "^5.0.0", diff --git a/frontend/playwright.config.ts b/frontend/playwright.config.ts new file mode 100644 index 0000000..0c42f7d --- /dev/null +++ b/frontend/playwright.config.ts @@ -0,0 +1,55 @@ +/// +import type { PlaywrightTestConfig } from '@playwright/test'; +import { devices } from '@playwright/test'; + +const config: PlaywrightTestConfig = { + testDir: './tests', + fullyParallel: false, + forbidOnly: !!process.env.CI, + retries: process.env.CI ? 2 : 0, + workers: 1, + reporter: 'html', + timeout: 10000, + expect: { + timeout: 10000 + }, + use: { + baseURL: 'http://localhost:3000', + trace: 'on-first-retry', + screenshot: 'only-on-failure', + actionTimeout: 10000 + }, + projects: [ + { + name: 'chromium', + use: { ...devices['Desktop Chrome'] } + }, + { + name: 'firefox', + use: { ...devices['Desktop Firefox'] } + }, + { + name: 'webkit', + use: { ...devices['Desktop Safari'] } + }, + { + name: 'mobile-chrome', + use: { ...devices['Pixel 5'] } + }, + { + name: 'mobile-safari', + use: { ...devices['iPhone 12'] } + } + ], + webServer: [ + { + command: 'cd .. && go run main.go > /dev/null 2>&1', + port: 3000, + reuseExistingServer: !process.env.CI, + stdout: 'pipe', + stderr: 'pipe' + } + ] +}; + +export default config; diff --git a/frontend/src/app.html b/frontend/src/app.html index 4a53236..05b6c02 100644 --- a/frontend/src/app.html +++ b/frontend/src/app.html @@ -9,37 +9,58 @@ /> %sveltekit.head% +
%sveltekit.body%
diff --git a/frontend/src/lib/components/Navigation.svelte b/frontend/src/lib/components/Navigation.svelte index 9150125..77f3478 100644 --- a/frontend/src/lib/components/Navigation.svelte +++ b/frontend/src/lib/components/Navigation.svelte @@ -23,7 +23,7 @@ {#if isMobile} -