# Reusable Select Component Usage This file tracks every page/partial that uses ``, ``, or ``. **Update this file whenever you add or remove a component from a view.** When modifying any component or its data source (`app/Data/Countries.php`), check all pages in the relevant section below and verify the change works correctly in each context. --- ## Data source **`app/Data/Countries.php`** — `App\Data\Countries` | Method | Used by component | |---|---| | `Countries::forPhoneCode()` | `` | | `Countries::forCountry()` | `` | | `Countries::forTimezone()` | `` | | `Countries::all()` | All three (via the above methods) | Adding or renaming a field in `Countries::all()` requires updating the corresponding `for*()` method too. --- ## Shared CSS / JS The `.csd-*` CSS rules and the `window.CSD` class are duplicated across all three component files inside `@once` guards. If you change the look or behaviour of the dropdown, **update all three component files**: - `resources/views/components/phone-code-select.blade.php` - `resources/views/components/country-select.blade.php` - `resources/views/components/timezone-select.blade.php` The `@once` Blade directive ensures the browser only receives one copy of the CSS/JS even when multiple components are on the same page. --- ## `` **File:** `resources/views/components/video-insights.blade.php` **Props:** `:video` — `Video` model instance. **Behaviour:** Renders the Insights tab panel (`
`), the drill-down modal, all `.ins-*` CSS, and all insights JS (`loadInsights`, `renderInsights`, modal openers, country/day/downloader drill-downs). Only renders if `Auth::id() === $video->user_id`. Must be placed **inside `.vdb-wrap`**, after the About panel, so the tab-switch CSS applies. The parent view must call `loadInsights()` (global, defined by this component) when the Insights tab is activated. **Data source:** `GET /videos/{video}/insights` (JSON) + drill-down routes `/insights/country/{code}`, `/insights/day/{date}`, `/insights/downloader/{userId}`. | View file | Placement | Notes | |---|---|---| | `resources/views/videos/partials/description-box.blade.php` | Inside `.vdb-wrap`, after About panel | Used by all three video type views (generic, match, music) | | `resources/views/videos/show.blade.php` | Inside `.vdb-wrap`, after About panel | Legacy view (not rendered by controller — kept in sync) | --- ## `` **File:** `resources/views/components/social-links-editor.blade.php` **Props:** `existing` — associative array keyed by platform name (e.g. `['twitter' => 'handle', 'whatsapp' => '97312345678']`). **Behaviour:** Dynamic add/remove rows; each row has a custom icon dropdown to pick the platform and a text input for the value. Supported platforms: `twitter`, `instagram`, `facebook`, `youtube`, `linkedin`, `tiktok`, `whatsapp`, `website`, `google_location`, `social_phone`, `social_email`. Hidden clear inputs ensure removed entries are cleared on save. Must be placed **inside a `
`**. **DB columns:** `twitter`, `instagram`, `facebook`, `youtube`, `linkedin`, `tiktok`, `website` (legacy), `whatsapp`, `google_location`, `social_phone`, `social_email`. | View file | Placement | Notes | |---|---|---| | `resources/views/user/profile.blade.php` | Social tab of Edit Profile modal | `$socialExisting` array passed from `@php` block above `@section('scripts')` | --- ## `` **File:** `resources/views/components/date-picker.blade.php` **Stored value:** `YYYY-MM-DD` string in a hidden input (same format as ``). **Props:** `name`, `id`, `value`, `label`, `required`, `class`, `style`, `minYear` (default 1900), `maxYear` (default current year). **Behaviour:** Day grid (5 columns, 1–31), month list (January–December), year searchable list (descending). Days auto-constrain on month/year change; invalid selected day resets automatically. | View file | Field name | Notes | |---|---|---| | `resources/views/user/profile.blade.php` | `birthday` | Replaces `` | | `resources/views/auth/register.blade.php` | `birthday` | Registration form — mandatory | --- ## `` **File:** `resources/views/components/image-cropper.blade.php` **Props:** `id` (unique, required), `width` (px, default 300), `height` (px, default 300), `shape` (`circle`|`square`, default `circle`), `folder` (storage subfolder), `filename` (base name without extension), `callback` (JS function name called with URL on success), `update-url` (endpoint to POST `{path}` after crop to update DB), `title` (modal heading). **Behaviour:** Renders a full-screen dark-themed modal with Cropme.js. Shows camera icon on avatar/banner hover (owner only). After crop: POSTs base64 to `/image-upload`, optionally POSTs the path to `update-url`, then calls `callback(url)`. Uses local assets (`public/js/cropme.min.js`, `public/css/cropme.min.css`). No jQuery required. **Assets needed:** `public/js/cropme.min.js`, `public/css/cropme.min.css`. **Routes needed:** `image.upload` (POST `/image-upload`). | View file | id / use | Notes | |---|---|---| | `resources/views/user/channel.blade.php` | `avatar` — circle 300×300 | Owner only; `update-url = profile.updateAvatar`; callback `onAvatarSaved` | | `resources/views/user/channel.blade.php` | `banner` — square 500×160 | Owner only; `update-url = profile.updateBanner`; callback `onBannerSaved` | | `resources/views/layouts/partials/upload-modal.blade.php` | `thumb_upload` — square 448×252 | Form mode; `target-input=thumbnail-modal`; output 1280px | | `resources/views/layouts/partials/edit-video-modal.blade.php` | `thumb_edit` — square 448×252 | Form mode; `target-input=edit-thumbnail-input`; output 1280px | | `resources/views/videos/create.blade.php` | `thumb_create_mobile` — square 448×252 | Mobile; `target-input=thumbnail`; output 1280px | | `resources/views/videos/edit.blade.php` | `thumb_edit_mobile` — square 448×252 | Mobile; `target-input=edit-thumbnail`; output 1280px | | `resources/views/playlists/index.blade.php` | `thumb_pl_create` — square 448×252 | Form mode; `target-input=playlist-thumbnail-input`; output 1280px | | `resources/views/playlists/show.blade.php` | `thumb_pl_edit` — square 448×252 | Form mode; `target-input=playlistThumbnailInput`; output 1280px | --- ## `` **File:** `resources/views/components/gender-select.blade.php` **Props:** `name`, `id`, `value` (ISO string: `"male"` or `"female"`), `label`, `required`, `class`, `style`. **Behaviour:** Custom dropdown with blue ♂ (male) and pink ♀ (female) symbols. No search needed. Stores value as `"male"` or `"female"`. | View file | Field name | Notes | |---|---|---| | `resources/views/auth/register.blade.php` | `gender` | Registration form — mandatory | --- ## `` Stored value format: `"+973|BH"` (dial_code + pipe + ISO2). To read only the dial code from a stored value: `explode('|', $value)[0]`. | View file | Field name | Notes | |---|---|---| | `resources/views/user/profile.blade.php` | `phone_code` | Paired with `phone_number` text input | --- ## `` Stored value: ISO2 code (e.g. `"BH"`). | View file | Field name | Notes | |---|---|---| | `resources/views/user/profile.blade.php` | `nationality` | Edit Profile form | | `resources/views/auth/register.blade.php` | `nationality` | Registration form — mandatory | --- ## `` Stored value: IANA timezone string (e.g. `"Asia/Bahrain"`). | View file | Field name | Notes | |---|---|---| | `resources/views/user/profile.blade.php` | `timezone` | Edit Profile form | --- ## Usage example ```blade {{-- Phone code + number side by side --}}
{{-- Country / nationality --}} {{-- Timezone --}} ``` --- ## Modification checklist When you modify any of these components, work through this list: - [ ] Update `app/Data/Countries.php` if the data structure changes - [ ] Update all three `.blade.php` component files if shared CSS/JS changes - [ ] Update the `for*()` method in `Countries.php` that feeds the changed component - [ ] Re-test every page listed in the usage tables above - [ ] Add/remove rows from the usage tables if views were added or removed