From 0b78c1cb54fcb6bf35d23e421e0dc15f507cf25d Mon Sep 17 00:00:00 2001 From: Fady Mondy Date: Wed, 27 Nov 2024 16:14:39 +0200 Subject: [PATCH] =?UTF-8?q?first=20commit=20=F0=9F=94=A5?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .github/CONTRIBUTING.md | 55 +++++++ .github/FUNDING.yml | 1 + .github/ISSUE_TEMPLATE/bug.yml | 66 +++++++++ .github/ISSUE_TEMPLATE/config.yaml | 11 ++ .github/SECURITY.md | 3 + .github/dependabot.yml | 12 ++ .github/workflows/dependabot-auto-merge.yml | 32 +++++ .github/workflows/fix-php-code-styling.yml | 30 ++++ .github/workflows/tests.yml | 68 +++++++++ .php-cs-fixer.dist.php | 37 +++++ 3x1io-tomato-support.md | 14 ++ CHANGELOG.md | 3 + CODE_OF_CONDUCT.md | 128 +++++++++++++++++ LICENSE.md | 21 +++ README.md | 69 +++++++++ SECURITY.md | 3 + arts/.gitkeep | 0 composer.json | 68 +++++++++ config/.gitkeep | 0 config/filament-support.php | 5 + ...2023_08_15_110333_create_tickets_table.php | 53 +++++++ ...15_110444_create_ticket_comments_table.php | 42 ++++++ module.json | 29 ++++ phpunit.xml | 26 ++++ pint.json | 14 ++ resources/lang/.gitkeep | 0 resources/lang/ar/messages.php | 5 + resources/lang/en/messages.php | 8 ++ src/Console/.gitkeep | 0 src/Console/FilamentSupportInstall.php | 44 ++++++ src/Filament/Resources/.gitkeep | 0 .../Resources/TicketCommentResource.php | 90 ++++++++++++ .../Pages/CreateTicketComment.php | 12 ++ .../Pages/EditTicketComment.php | 19 +++ .../Pages/ListTicketComments.php | 19 +++ src/Filament/Resources/TicketResource.php | 134 ++++++++++++++++++ .../TicketResource/Pages/CreateTicket.php | 12 ++ .../TicketResource/Pages/EditTicket.php | 19 +++ .../TicketResource/Pages/ListTickets.php | 19 +++ src/FilamentSupportPlugin.php | 30 ++++ src/FilamentSupportServiceProvider.php | 47 ++++++ src/Models/.gitkeep | 0 src/Models/Ticket.php | 53 +++++++ src/Models/TicketComment.php | 31 ++++ testbench.yaml | 24 ++++ tests/Pest.php | 5 + tests/database/database.sqlite | Bin 0 -> 245760 bytes tests/database/factories/.gitkeep | 0 tests/database/factories/UserFactory.php | 23 +++ tests/src/AdminPanelProvider.php | 50 +++++++ tests/src/DebugTest.php | 5 + tests/src/Models/.gitkeep | 0 tests/src/Models/User.php | 34 +++++ tests/src/TestCase.php | 53 +++++++ 54 files changed, 1526 insertions(+) create mode 100644 .github/CONTRIBUTING.md create mode 100644 .github/FUNDING.yml create mode 100644 .github/ISSUE_TEMPLATE/bug.yml create mode 100644 .github/ISSUE_TEMPLATE/config.yaml create mode 100644 .github/SECURITY.md create mode 100644 .github/dependabot.yml create mode 100644 .github/workflows/dependabot-auto-merge.yml create mode 100644 .github/workflows/fix-php-code-styling.yml create mode 100644 .github/workflows/tests.yml create mode 100644 .php-cs-fixer.dist.php create mode 100644 3x1io-tomato-support.md create mode 100644 CHANGELOG.md create mode 100644 CODE_OF_CONDUCT.md create mode 100644 LICENSE.md create mode 100644 README.md create mode 100644 SECURITY.md create mode 100644 arts/.gitkeep create mode 100644 composer.json create mode 100644 config/.gitkeep create mode 100644 config/filament-support.php create mode 100644 database/migrations/2023_08_15_110333_create_tickets_table.php create mode 100644 database/migrations/2023_08_15_110444_create_ticket_comments_table.php create mode 100644 module.json create mode 100644 phpunit.xml create mode 100644 pint.json create mode 100644 resources/lang/.gitkeep create mode 100644 resources/lang/ar/messages.php create mode 100644 resources/lang/en/messages.php create mode 100644 src/Console/.gitkeep create mode 100644 src/Console/FilamentSupportInstall.php create mode 100644 src/Filament/Resources/.gitkeep create mode 100644 src/Filament/Resources/TicketCommentResource.php create mode 100644 src/Filament/Resources/TicketCommentResource/Pages/CreateTicketComment.php create mode 100644 src/Filament/Resources/TicketCommentResource/Pages/EditTicketComment.php create mode 100644 src/Filament/Resources/TicketCommentResource/Pages/ListTicketComments.php create mode 100644 src/Filament/Resources/TicketResource.php create mode 100644 src/Filament/Resources/TicketResource/Pages/CreateTicket.php create mode 100644 src/Filament/Resources/TicketResource/Pages/EditTicket.php create mode 100644 src/Filament/Resources/TicketResource/Pages/ListTickets.php create mode 100644 src/FilamentSupportPlugin.php create mode 100644 src/FilamentSupportServiceProvider.php create mode 100644 src/Models/.gitkeep create mode 100644 src/Models/Ticket.php create mode 100644 src/Models/TicketComment.php create mode 100644 testbench.yaml create mode 100644 tests/Pest.php create mode 100644 tests/database/database.sqlite create mode 100644 tests/database/factories/.gitkeep create mode 100644 tests/database/factories/UserFactory.php create mode 100644 tests/src/AdminPanelProvider.php create mode 100644 tests/src/DebugTest.php create mode 100644 tests/src/Models/.gitkeep create mode 100644 tests/src/Models/User.php create mode 100644 tests/src/TestCase.php diff --git a/.github/CONTRIBUTING.md b/.github/CONTRIBUTING.md new file mode 100644 index 0000000..b0ee5d8 --- /dev/null +++ b/.github/CONTRIBUTING.md @@ -0,0 +1,55 @@ +# Contributing + +Contributions are **welcome** and will be fully **credited**. + +Please read and understand the contribution guide before creating an issue or pull request. + +## Etiquette + +This project is open source, and as such, the maintainers give their free time to build and maintain the source code +held within. They make the code freely available in the hope that it will be of use to other developers. It would be +extremely unfair for them to suffer abuse or anger for their hard work. + +Please be considerate towards maintainers when raising issues or presenting pull requests. Let's show the +world that developers are civilized and selfless people. + +It's the duty of the maintainer to ensure that all submissions to the project are of sufficient +quality to benefit the project. Many developers have different skills, strengths, and weaknesses. Respect the maintainer's decision, and do not be upset or abusive if your submission is not used. + +## Viability + +When requesting or submitting new features, first consider whether it might be useful to others. Open +source projects are used by many developers, who may have entirely different needs to your own. Think about +whether or not your feature is likely to be used by other users of the project. + +## Procedure + +Before filing an issue: + +- Attempt to replicate the problem, to ensure that it wasn't a coincidental incident. +- Check to make sure your feature suggestion isn't already present within the project. +- Check the pull requests tab to ensure that the bug doesn't have a fix in progress. +- Check the pull requests tab to ensure that the feature isn't already in progress. + +Before submitting a pull request: + +- Check the codebase to ensure that your feature doesn't already exist. +- Check the pull requests to ensure that another person hasn't already submitted the feature or fix. + +## Requirements + +If the project maintainer has any additional requirements, you will find them listed here. + +- **[PSR-2 Coding Standard](https://github.com/php-fig/fig-standards/blob/master/accepted/PSR-2-coding-style-guide.md)** - The easiest way to apply the conventions is to install [PHP Code Sniffer](https://pear.php.net/package/PHP_CodeSniffer). + +- **Add tests!** - Your patch won't be accepted if it doesn't have tests. + +- **Document any change in behaviour** - Make sure the `README.md` and any other relevant documentation are kept up-to-date. + +- **Consider our release cycle** - We try to follow [SemVer v2.0.0](https://semver.org/). Randomly breaking public APIs is not an option. + +- **One pull request per feature** - If you want to do more than one thing, send multiple pull requests. + +- **Send coherent history** - Make sure each individual commit in your pull request is meaningful. If you had to make multiple intermediate commits while developing, please [squash them](https://www.git-scm.com/book/en/v2/Git-Tools-Rewriting-History#Changing-Multiple-Commit-Messages) before submitting. + +**Happy coding**! diff --git a/.github/FUNDING.yml b/.github/FUNDING.yml new file mode 100644 index 0000000..892ba05 --- /dev/null +++ b/.github/FUNDING.yml @@ -0,0 +1 @@ +github: [3x1io] diff --git a/.github/ISSUE_TEMPLATE/bug.yml b/.github/ISSUE_TEMPLATE/bug.yml new file mode 100644 index 0000000..8fa85ce --- /dev/null +++ b/.github/ISSUE_TEMPLATE/bug.yml @@ -0,0 +1,66 @@ +name: Bug Report +description: Report an Issue or Bug with the Package +title: "[Bug]: " +labels: ["bug"] +body: + - type: markdown + attributes: + value: | + We're sorry to hear you have a problem. Can you help us solve it by providing the following details. + - type: textarea + id: what-happened + attributes: + label: What happened? + description: What did you expect to happen? + placeholder: I cannot currently do X thing because when I do, it breaks X thing. + validations: + required: true + - type: textarea + id: how-to-reproduce + attributes: + label: How to reproduce the bug + description: How did this occur, please add any config values used and provide a set of reliable steps if possible. + placeholder: When I do X I see Y. + validations: + required: true + - type: input + id: package-version + attributes: + label: Package Version + description: What version of our Package are you running? Please be as specific as possible + placeholder: 2.0.0 + validations: + required: true + - type: input + id: php-version + attributes: + label: PHP Version + description: What version of PHP are you running? Please be as specific as possible + placeholder: 8.2.0 + validations: + required: true + - type: input + id: laravel-version + attributes: + label: Laravel Version + description: What version of Laravel are you running? Please be as specific as possible + placeholder: 9.0.0 + validations: + required: true + - type: dropdown + id: operating-systems + attributes: + label: Which operating systems does with happen with? + description: You may select more than one. + multiple: true + options: + - macOS + - Windows + - Linux + - type: textarea + id: notes + attributes: + label: Notes + description: Use this field to provide any other notes that you feel might be relevant to the issue. + validations: + required: false diff --git a/.github/ISSUE_TEMPLATE/config.yaml b/.github/ISSUE_TEMPLATE/config.yaml new file mode 100644 index 0000000..a1dbb37 --- /dev/null +++ b/.github/ISSUE_TEMPLATE/config.yaml @@ -0,0 +1,11 @@ +blank_issues_enabled: false +contact_links: + - name: Ask a question + url: https://github.com/tomatophp/filament-support/discussions/new?category=q-a + about: Ask the community for help + - name: Request a feature + url: https://github.com/tomatophp/filament-support/discussions/new?category=ideas + about: Share ideas for new features + - name: Report a security issue + url: https://github.com/tomatophp/filament-support/security/policy + about: Learn how to notify us for sensitive bugs diff --git a/.github/SECURITY.md b/.github/SECURITY.md new file mode 100644 index 0000000..b2490a9 --- /dev/null +++ b/.github/SECURITY.md @@ -0,0 +1,3 @@ +# Security Policy + +If you discover any security related issues, please email info@3x1.io instead of using the issue tracker. diff --git a/.github/dependabot.yml b/.github/dependabot.yml new file mode 100644 index 0000000..0bc378d --- /dev/null +++ b/.github/dependabot.yml @@ -0,0 +1,12 @@ +# Please see the documentation for all configuration options: +# https://help.github.com/github/administering-a-repository/configuration-options-for-dependency-updates + +version: 2 +updates: + + - package-ecosystem: "github-actions" + directory: "/" + schedule: + interval: "weekly" + labels: + - "dependencies" diff --git a/.github/workflows/dependabot-auto-merge.yml b/.github/workflows/dependabot-auto-merge.yml new file mode 100644 index 0000000..27c23a4 --- /dev/null +++ b/.github/workflows/dependabot-auto-merge.yml @@ -0,0 +1,32 @@ +name: dependabot-auto-merge +on: pull_request_target + +permissions: + pull-requests: write + contents: write + +jobs: + dependabot: + runs-on: ubuntu-latest + if: ${{ github.actor == 'dependabot[bot]' }} + steps: + + - name: Dependabot metadata + id: metadata + uses: dependabot/fetch-metadata@v2.2.0 + with: + github-token: "${{ secrets.GITHUB_TOKEN }}" + + - name: Auto-merge Dependabot PRs for semver-minor updates + if: ${{steps.metadata.outputs.update-type == 'version-update:semver-minor'}} + run: gh pr merge --auto --merge "$PR_URL" + env: + PR_URL: ${{github.event.pull_request.html_url}} + GITHUB_TOKEN: ${{secrets.GITHUB_TOKEN}} + + - name: Auto-merge Dependabot PRs for semver-patch updates + if: ${{steps.metadata.outputs.update-type == 'version-update:semver-patch'}} + run: gh pr merge --auto --merge "$PR_URL" + env: + PR_URL: ${{github.event.pull_request.html_url}} + GITHUB_TOKEN: ${{secrets.GITHUB_TOKEN}} diff --git a/.github/workflows/fix-php-code-styling.yml b/.github/workflows/fix-php-code-styling.yml new file mode 100644 index 0000000..e71024d --- /dev/null +++ b/.github/workflows/fix-php-code-styling.yml @@ -0,0 +1,30 @@ +name: 'PHP Code Styling' + +on: + workflow_dispatch: + push: + branches: + - master + paths: + - '**.php' + +permissions: + contents: write + +jobs: + lint: + runs-on: ubuntu-latest + steps: + - name: Checkout code + uses: actions/checkout@v4 + with: + ref: ${{ github.head_ref }} + + - name: Fix PHP code style issues + uses: aglipanci/laravel-pint-action@v2 + + - name: Commit changes + uses: stefanzweifel/git-auto-commit-action@v5 + with: + commit_message: "Format Code" + commit_user_name: 'GitHub Actions' diff --git a/.github/workflows/tests.yml b/.github/workflows/tests.yml new file mode 100644 index 0000000..14a3ed3 --- /dev/null +++ b/.github/workflows/tests.yml @@ -0,0 +1,68 @@ +name: "Tests" + +on: + workflow_dispatch: + push: + branches: + - master + paths: + - '**.php' + pull_request: + types: + - opened + - synchronize + branches: + - master + paths: + - '**.php' + - '.github/workflows/tests.yml' + - 'phpunit.xml.dist' + - 'composer.json' + - 'composer.lock' + +jobs: + test: + runs-on: ${{ matrix.os }} + strategy: + fail-fast: true + matrix: + os: [ubuntu-latest] + php: [8.3, 8.2] + laravel: [11.*] + stability: [prefer-stable] + include: + - laravel: 11.* + testbench: 9.* + carbon: 3.* + collision: 8.* + name: P${{ matrix.php }} - L${{ matrix.laravel }} - ${{ matrix.stability }} - ${{ matrix.os }} + steps: + - name: Checkout Code + uses: actions/checkout@v4 + + - name: Cache Dependencies + uses: actions/cache@v4 + with: + path: ~/.composer/cache/files + key: dependencies-laravel-${{ matrix.laravel }}-php-${{ matrix.php }}-composer-${{ hashFiles('composer.json') }} + + - name: Setup PHP + uses: shivammathur/setup-php@v2 + with: + php-version: ${{ matrix.php }} + extensions: dom, curl, libxml, mbstring, zip, pcntl, pdo, sqlite, pdo_sqlite, bcmath, soap, intl, gd, exif, iconv, imagick, fileinfo + coverage: none + + - name: Install Dependencies + run: | + echo "::add-matcher::${{ runner.tool_cache }}/php.json" + echo "::add-matcher::${{ runner.tool_cache }}/phpunit.json" + + - name: Install Dependencies + run: | + composer require "laravel/framework:${{ matrix.laravel }}" "orchestra/testbench:${{ matrix.testbench }}" "nesbot/carbon:${{ matrix.carbon }}" "nunomaduro/collision:${{ matrix.collision }}" --no-interaction --no-update + composer update --${{ matrix.stability }} --prefer-dist --no-interaction + composer db + + - name: Execute tests + run: vendor/bin/pest diff --git a/.php-cs-fixer.dist.php b/.php-cs-fixer.dist.php new file mode 100644 index 0000000..4123157 --- /dev/null +++ b/.php-cs-fixer.dist.php @@ -0,0 +1,37 @@ +notPath('bootstrap/*') + ->notPath('storage/*') + ->notPath('resources/view/mail/*') + ->in([ + __DIR__ . '/src', + __DIR__ . '/tests', + ]) + ->name('*.php') + ->notName('*.blade.php') + ->ignoreDotFiles(true) + ->ignoreVCS(true); + +return (new PhpCsFixer\Config()) + ->setRules([ + '@PSR2' => true, + 'array_syntax' => ['syntax' => 'short'], + 'ordered_imports' => ['sort_algorithm' => 'alpha'], + 'no_unused_imports' => true, + 'not_operator_with_successor_space' => true, + 'trailing_comma_in_multiline' => true, + 'phpdoc_scalar' => true, + 'unary_operator_spaces' => true, + 'binary_operator_spaces' => true, + 'blank_line_before_statement' => [ + 'statements' => ['break', 'continue', 'declare', 'return', 'throw', 'try'], + ], + 'phpdoc_single_line_var_spacing' => true, + 'phpdoc_var_without_name' => true, + 'method_argument_space' => [ + 'on_multiline' => 'ensure_fully_multiline', + 'keep_multiple_spaces_after_comma' => true, + ] + ]) + ->setFinder($finder); diff --git a/3x1io-tomato-support.md b/3x1io-tomato-support.md new file mode 100644 index 0000000..56043a4 --- /dev/null +++ b/3x1io-tomato-support.md @@ -0,0 +1,14 @@ +--- +name: Customers Support +slug: 3x1io-tomato-support +author_slug: 3x1io +categories: [developer-tools] +description: Manage your customers contact us form and customer support ticking system for FilamentPHP +discord_url: +docs_url: https://raw.githubusercontent.com/tomatophp/filament-support/master/README.md +github_repository: tomatophp/filament-support +has_dark_theme: true +has_translations: true +versions: [3] +publish_date: 2024-11-27 +--- diff --git a/CHANGELOG.md b/CHANGELOG.md new file mode 100644 index 0000000..e0a5086 --- /dev/null +++ b/CHANGELOG.md @@ -0,0 +1,3 @@ +# V1.0.0 + +First release of the package diff --git a/CODE_OF_CONDUCT.md b/CODE_OF_CONDUCT.md new file mode 100644 index 0000000..18c9147 --- /dev/null +++ b/CODE_OF_CONDUCT.md @@ -0,0 +1,128 @@ +# Contributor Covenant Code of Conduct + +## Our Pledge + +We as members, contributors, and leaders pledge to make participation in our +community a harassment-free experience for everyone, regardless of age, body +size, visible or invisible disability, ethnicity, sex characteristics, gender +identity and expression, level of experience, education, socio-economic status, +nationality, personal appearance, race, religion, or sexual identity +and orientation. + +We pledge to act and interact in ways that contribute to an open, welcoming, +diverse, inclusive, and healthy community. + +## Our Standards + +Examples of behavior that contributes to a positive environment for our +community include: + +* Demonstrating empathy and kindness toward other people +* Being respectful of differing opinions, viewpoints, and experiences +* Giving and gracefully accepting constructive feedback +* Accepting responsibility and apologizing to those affected by our mistakes, + and learning from the experience +* Focusing on what is best not just for us as individuals, but for the + overall community + +Examples of unacceptable behavior include: + +* The use of sexualized language or imagery, and sexual attention or + advances of any kind +* Trolling, insulting or derogatory comments, and personal or political attacks +* Public or private harassment +* Publishing others' private information, such as a physical or email + address, without their explicit permission +* Other conduct which could reasonably be considered inappropriate in a + professional setting + +## Enforcement Responsibilities + +Community leaders are responsible for clarifying and enforcing our standards of +acceptable behavior and will take appropriate and fair corrective action in +response to any behavior that they deem inappropriate, threatening, offensive, +or harmful. + +Community leaders have the right and responsibility to remove, edit, or reject +comments, commits, code, wiki edits, issues, and other contributions that are +not aligned to this Code of Conduct, and will communicate reasons for moderation +decisions when appropriate. + +## Scope + +This Code of Conduct applies within all community spaces, and also applies when +an individual is officially representing the community in public spaces. +Examples of representing our community include using an official e-mail address, +posting via an official social media account, or acting as an appointed +representative at an online or offline event. + +## Enforcement + +Instances of abusive, harassing, or otherwise unacceptable behavior may be +reported to the community leaders responsible for enforcement at +. +All complaints will be reviewed and investigated promptly and fairly. + +All community leaders are obligated to respect the privacy and security of the +reporter of any incident. + +## Enforcement Guidelines + +Community leaders will follow these Community Impact Guidelines in determining +the consequences for any action they deem in violation of this Code of Conduct: + +### 1. Correction + +**Community Impact**: Use of inappropriate language or other behavior deemed +unprofessional or unwelcome in the community. + +**Consequence**: A private, written warning from community leaders, providing +clarity around the nature of the violation and an explanation of why the +behavior was inappropriate. A public apology may be requested. + +### 2. Warning + +**Community Impact**: A violation through a single incident or series +of actions. + +**Consequence**: A warning with consequences for continued behavior. No +interaction with the people involved, including unsolicited interaction with +those enforcing the Code of Conduct, for a specified period of time. This +includes avoiding interactions in community spaces as well as external channels +like social media. Violating these terms may lead to a temporary or +permanent ban. + +### 3. Temporary Ban + +**Community Impact**: A serious violation of community standards, including +sustained inappropriate behavior. + +**Consequence**: A temporary ban from any sort of interaction or public +communication with the community for a specified period of time. No public or +private interaction with the people involved, including unsolicited interaction +with those enforcing the Code of Conduct, is allowed during this period. +Violating these terms may lead to a permanent ban. + +### 4. Permanent Ban + +**Community Impact**: Demonstrating a pattern of violation of community +standards, including sustained inappropriate behavior, harassment of an +individual, or aggression toward or disparagement of classes of individuals. + +**Consequence**: A permanent ban from any sort of public interaction within +the community. + +## Attribution + +This Code of Conduct is adapted from the [Contributor Covenant][homepage], +version 2.0, available at +https://www.contributor-covenant.org/version/2/0/code_of_conduct.html. + +Community Impact Guidelines were inspired by [Mozilla's code of conduct +enforcement ladder](https://github.com/mozilla/diversity). + +[homepage]: https://www.contributor-covenant.org + +For answers to common questions about this code of conduct, see the FAQ at +https://www.contributor-covenant.org/faq. Translations are available at +https://www.contributor-covenant.org/translations. diff --git a/LICENSE.md b/LICENSE.md new file mode 100644 index 0000000..e66364e --- /dev/null +++ b/LICENSE.md @@ -0,0 +1,21 @@ +The MIT License (MIT) + +Copyright (c) + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +THE SOFTWARE. diff --git a/README.md b/README.md new file mode 100644 index 0000000..13ebdac --- /dev/null +++ b/README.md @@ -0,0 +1,69 @@ +![Screenshot](https://raw.githubusercontent.com/tomatophp/filament-support/master/art/screenshot.jpg) + +# Filament support + +[![Latest Stable Version](https://poser.pugx.org/tomatophp/filament-support/version.svg)](https://packagist.org/packages/tomatophp/filament-support) +[![License](https://poser.pugx.org/tomatophp/filament-support/license.svg)](https://packagist.org/packages/tomatophp/filament-support) +[![Downloads](https://poser.pugx.org/tomatophp/filament-support/d/total.svg)](https://packagist.org/packages/tomatophp/filament-support) + +Manage your customers contact us form and customer support ticking system for FilamentPHP + +## Installation + +```bash +composer require tomatophp/filament-support +``` +after install your package please run this command + +```bash +php artisan filament-support:install +``` + +finally register the plugin on `/app/Providers/Filament/AdminPanelProvider.php` + +```php +->plugin(\TomatoPHP\FilamentSupport\FilamentSupportPlugin::make()) +``` + + +## Publish Assets + +you can publish config file by use this command + +```bash +php artisan vendor:publish --tag="filament-support-config" +``` + +you can publish views file by use this command + +```bash +php artisan vendor:publish --tag="filament-support-views" +``` + +you can publish languages file by use this command + +```bash +php artisan vendor:publish --tag="filament-support-lang" +``` + +you can publish migrations file by use this command + +```bash +php artisan vendor:publish --tag="filament-support-migrations" +``` + +## Changelog + +Please see [CHANGELOG](CHANGELOG.md) for more information on what has changed recently. + +## Security + +Please see [SECURITY](SECURITY.md) for more information about security. + +## Credits + +- [Fady Mondy](mailto:info@3x1.io) + +## License + +The MIT License (MIT). Please see [License File](LICENSE.md) for more information. diff --git a/SECURITY.md b/SECURITY.md new file mode 100644 index 0000000..b2490a9 --- /dev/null +++ b/SECURITY.md @@ -0,0 +1,3 @@ +# Security Policy + +If you discover any security related issues, please email info@3x1.io instead of using the issue tracker. diff --git a/arts/.gitkeep b/arts/.gitkeep new file mode 100644 index 0000000..e69de29 diff --git a/composer.json b/composer.json new file mode 100644 index 0000000..7f1be97 --- /dev/null +++ b/composer.json @@ -0,0 +1,68 @@ +{ + "name": "tomatophp/filament-support", + "type": "library", + "description": "Manage your customers contact us form and customer support ticking system for FilamentPHP", + "keywords": [ + "php", + "laravel", + "template" + ], + "license": "MIT", + "autoload": { + "psr-4": { + "TomatoPHP\\FilamentSupport\\": "src/" + } + }, + "autoload-dev": { + "psr-4": { + "TomatoPHP\\FilamentSupport\\Tests\\": "tests/src/", + "TomatoPHP\\FilamentSupport\\Tests\\Database\\Factories\\": "tests/database/factories" + } + }, + "extra": { + "laravel": { + "providers": [ + "TomatoPHP\\FilamentSupport\\FilamentSupportServiceProvider" + ] + } + }, + "authors": [ + { + "name": "Fady Mondy", + "email": "info@3x1.io" + } + ], + "config": { + "sort-packages": true, + "allow-plugins": { + "pestphp/pest-plugin": true, + "phpstan/extension-installer": true + } + }, + "scripts": { + "testbench": "vendor/bin/testbench package:discover --ansi", + "db": "vendor/bin/testbench package:create-sqlite-db && vendor/bin/testbench migrate", + "analyse": "vendor/bin/phpstan analyse src tests", + "test": "vendor/bin/pest", + "test-coverage": "vendor/bin/pest --coverage", + "format": "vendor/bin/pint" + }, + "require": { + "php": "^8.1|^8.2", + "tomatophp/console-helpers": "^1.1", + "filament/filament": "^3.2" + }, + "require-dev": { + "laravel/pint": "^1.18", + "livewire/livewire": "^2.10|^3.0", + "nunomaduro/larastan": "^2.9", + "orchestra/testbench": "^9.5", + "pestphp/pest": "^2.36", + "pestphp/pest-plugin-laravel": "^2.4", + "pestphp/pest-plugin-livewire": "^2.1", + "phpstan/extension-installer": "^1.4", + "phpstan/phpstan-deprecation-rules": "^1.2", + "phpstan/phpstan-phpunit": "^1.4" + }, + "version": "v1.0.0" +} diff --git a/config/.gitkeep b/config/.gitkeep new file mode 100644 index 0000000..e69de29 diff --git a/config/filament-support.php b/config/filament-support.php new file mode 100644 index 0000000..f5bf951 --- /dev/null +++ b/config/filament-support.php @@ -0,0 +1,5 @@ +id(); + + //Add Status For Tickets + $table->string('status')->nullable()->default('pending'); + + //Link User To Ticket With Morph + $table->unsignedBigInteger('user_id'); + $table->string('user_type'); + + //Add User For Tickets + $table->string('name')->nullable(); + $table->string('phone')->nullable(); + + //Add User For Tickets + $table->string('subject'); + $table->string('code')->unique()->index(); + + $table->longText('message')->nullable(); + + //Get Last update time + $table->timestamp('last_update')->nullable(); + + //Is closed + $table->boolean('is_closed')->default(false)->nullable(); + + $table->timestamps(); + }); + } + } + + /** + * Reverse the migrations. + */ + public function down(): void + { + Schema::dropIfExists('tickets'); + } +}; diff --git a/database/migrations/2023_08_15_110444_create_ticket_comments_table.php b/database/migrations/2023_08_15_110444_create_ticket_comments_table.php new file mode 100644 index 0000000..a4f4de3 --- /dev/null +++ b/database/migrations/2023_08_15_110444_create_ticket_comments_table.php @@ -0,0 +1,42 @@ +id(); + + //Ref + $table->foreignId('ticket_id')->constrained('tickets')->onDelete('cascade'); + + //Link User To Ticket With Morph + $table->unsignedBigInteger('user_id'); + $table->string('user_type'); + + //Add User For Tickets + $table->longText('response'); + + $table->timestamps(); + }); + } + } + + /** + * Reverse the migrations. + */ + public function down(): void + { + Schema::disableForeignKeyConstraints(); + Schema::dropIfExists('ticket_comments'); + Schema::enableForeignKeyConstraints(); + } +}; diff --git a/module.json b/module.json new file mode 100644 index 0000000..6c41b68 --- /dev/null +++ b/module.json @@ -0,0 +1,29 @@ +{ + "name": "FilamentSupport", + "alias": "filament-support", + "description": { + "ar": "Manage your customers contact us form and customer support ticking system for FilamentPHP", + "en": "Manage your customers contact us form and customer support ticking system for FilamentPHP", + "gr": "Manage your customers contact us form and customer support ticking system for FilamentPHP", + "sp": "Manage your customers contact us form and customer support ticking system for FilamentPHP" + }, + "keywords": [], + "priority": 0, + "providers": [ + "TomatoPHP\\FilamentSupport\\FilamentSupportServiceProvider" + ], + "files": [], + "title": { + "ar": "Filament support", + "en": "Filament support", + "gr": "Filament support", + "sp": "Filament support" + }, + "color": "#cc1448", + "icon": "heroicon-c-users", + "placeholder": "https://raw.githubusercontent.com/tomatophp/filament-support/master/art/screenshot.jpg", + "type": "plugin", + "version": "v1.0.0", + "github" : "https://github.com/tomatophp/filament-support", + "docs" : "https://github.com/tomatophp/filament-support" +} diff --git a/phpunit.xml b/phpunit.xml new file mode 100644 index 0000000..e542661 --- /dev/null +++ b/phpunit.xml @@ -0,0 +1,26 @@ + + + + + ./tests/ + + + + + ./src + + + + + + + diff --git a/pint.json b/pint.json new file mode 100644 index 0000000..c6ddb49 --- /dev/null +++ b/pint.json @@ -0,0 +1,14 @@ +{ + "preset": "laravel", + "rules": { + "blank_line_before_statement": true, + "concat_space": { + "spacing": "one" + }, + "method_argument_space": true, + "single_trait_insert_per_statement": true, + "types_spaces": { + "space": "single" + } + } +} diff --git a/resources/lang/.gitkeep b/resources/lang/.gitkeep new file mode 100644 index 0000000..e69de29 diff --git a/resources/lang/ar/messages.php b/resources/lang/ar/messages.php new file mode 100644 index 0000000..ca5d8ed --- /dev/null +++ b/resources/lang/ar/messages.php @@ -0,0 +1,5 @@ + [ + "title" => "Tickets", + "single" => "Ticket" + ] +]; diff --git a/src/Console/.gitkeep b/src/Console/.gitkeep new file mode 100644 index 0000000..e69de29 diff --git a/src/Console/FilamentSupportInstall.php b/src/Console/FilamentSupportInstall.php new file mode 100644 index 0000000..dd04101 --- /dev/null +++ b/src/Console/FilamentSupportInstall.php @@ -0,0 +1,44 @@ +info('Publish Vendor Assets'); + $this->artisanCommand(["migrate"]); + $this->artisanCommand(["optimize:clear"]); + $this->info('Filament support installed successfully.'); + } +} diff --git a/src/Filament/Resources/.gitkeep b/src/Filament/Resources/.gitkeep new file mode 100644 index 0000000..e69de29 diff --git a/src/Filament/Resources/TicketCommentResource.php b/src/Filament/Resources/TicketCommentResource.php new file mode 100644 index 0000000..fec90d8 --- /dev/null +++ b/src/Filament/Resources/TicketCommentResource.php @@ -0,0 +1,90 @@ +schema([ + Forms\Components\TextInput::make('ticket_id') + ->required() + ->numeric(), + Forms\Components\TextInput::make('user_id') + ->required() + ->numeric(), + Forms\Components\TextInput::make('user_type') + ->required() + ->maxLength(255), + Forms\Components\Textarea::make('response') + ->required() + ->columnSpanFull(), + ]); + } + + public static function table(Table $table): Table + { + return $table + ->columns([ + Tables\Columns\TextColumn::make('ticket_id') + ->numeric() + ->sortable(), + Tables\Columns\TextColumn::make('user_id') + ->numeric() + ->sortable(), + Tables\Columns\TextColumn::make('user_type') + ->searchable(), + Tables\Columns\TextColumn::make('created_at') + ->dateTime() + ->sortable() + ->toggleable(isToggledHiddenByDefault: true), + Tables\Columns\TextColumn::make('updated_at') + ->dateTime() + ->sortable() + ->toggleable(isToggledHiddenByDefault: true), + ]) + ->filters([ + // + ]) + ->actions([ + Tables\Actions\EditAction::make(), + ]) + ->bulkActions([ + Tables\Actions\BulkActionGroup::make([ + Tables\Actions\DeleteBulkAction::make(), + ]), + ]); + } + + public static function getRelations(): array + { + return [ + // + ]; + } + + public static function getPages(): array + { + return [ + 'index' => Pages\ListTicketComments::route('/'), + 'create' => Pages\CreateTicketComment::route('/create'), + 'edit' => Pages\EditTicketComment::route('/{record}/edit'), + ]; + } +} diff --git a/src/Filament/Resources/TicketCommentResource/Pages/CreateTicketComment.php b/src/Filament/Resources/TicketCommentResource/Pages/CreateTicketComment.php new file mode 100644 index 0000000..86787ab --- /dev/null +++ b/src/Filament/Resources/TicketCommentResource/Pages/CreateTicketComment.php @@ -0,0 +1,12 @@ +schema([ + Forms\Components\TextInput::make('status') + ->maxLength(255) + ->default('pending'), + Forms\Components\TextInput::make('user_id') + ->required() + ->numeric(), + Forms\Components\TextInput::make('user_type') + ->required() + ->maxLength(255), + Forms\Components\TextInput::make('name') + ->maxLength(255), + Forms\Components\TextInput::make('phone') + ->tel() + ->maxLength(255), + Forms\Components\TextInput::make('subject') + ->required() + ->maxLength(255), + Forms\Components\TextInput::make('code') + ->required() + ->maxLength(255), + Forms\Components\Textarea::make('message') + ->columnSpanFull(), + Forms\Components\DateTimePicker::make('last_update'), + Forms\Components\Toggle::make('is_closed'), + ]); + } + + public static function table(Table $table): Table + { + return $table + ->columns([ + Tables\Columns\TextColumn::make('status') + ->searchable(), + Tables\Columns\TextColumn::make('user_id') + ->numeric() + ->sortable(), + Tables\Columns\TextColumn::make('user_type') + ->searchable(), + Tables\Columns\TextColumn::make('name') + ->searchable(), + Tables\Columns\TextColumn::make('phone') + ->searchable(), + Tables\Columns\TextColumn::make('subject') + ->searchable(), + Tables\Columns\TextColumn::make('code') + ->searchable(), + Tables\Columns\TextColumn::make('last_update') + ->dateTime() + ->sortable(), + Tables\Columns\IconColumn::make('is_closed') + ->boolean(), + Tables\Columns\TextColumn::make('created_at') + ->dateTime() + ->sortable() + ->toggleable(isToggledHiddenByDefault: true), + Tables\Columns\TextColumn::make('updated_at') + ->dateTime() + ->sortable() + ->toggleable(isToggledHiddenByDefault: true), + ]) + ->filters([ + // + ]) + ->actions([ + Tables\Actions\EditAction::make(), + ]) + ->bulkActions([ + Tables\Actions\BulkActionGroup::make([ + Tables\Actions\DeleteBulkAction::make(), + ]), + ]); + } + + public static function getRelations(): array + { + return [ + // + ]; + } + + public static function getPages(): array + { + return [ + 'index' => Pages\ListTickets::route('/'), + 'create' => Pages\CreateTicket::route('/create'), + 'edit' => Pages\EditTicket::route('/{record}/edit'), + ]; + } +} diff --git a/src/Filament/Resources/TicketResource/Pages/CreateTicket.php b/src/Filament/Resources/TicketResource/Pages/CreateTicket.php new file mode 100644 index 0000000..5bb30b2 --- /dev/null +++ b/src/Filament/Resources/TicketResource/Pages/CreateTicket.php @@ -0,0 +1,12 @@ +commands([ + \TomatoPHP\FilamentSupport\Console\FilamentSupportInstall::class, + ]); + + //Register Config file + $this->mergeConfigFrom(__DIR__.'/../config/filament-support.php', 'filament-support'); + + //Publish Config + $this->publishes([ + __DIR__.'/../config/filament-support.php' => config_path('filament-support.php'), + ], 'filament-support-config'); + + //Register Migrations + $this->loadMigrationsFrom(__DIR__.'/../database/migrations'); + + //Publish Migrations + $this->publishes([ + __DIR__.'/../database/migrations' => database_path('migrations'), + ], 'filament-support-migrations'); + + + //Register Langs + $this->loadTranslationsFrom(__DIR__.'/../resources/lang', 'filament-support'); + + //Publish Lang + $this->publishes([ + __DIR__.'/../resources/lang' => base_path('lang/vendor/filament-support'), + ], 'filament-support-lang'); + } + + public function boot(): void + { + //you boot methods here + } +} diff --git a/src/Models/.gitkeep b/src/Models/.gitkeep new file mode 100644 index 0000000..e69de29 diff --git a/src/Models/Ticket.php b/src/Models/Ticket.php new file mode 100644 index 0000000..414dac9 --- /dev/null +++ b/src/Models/Ticket.php @@ -0,0 +1,53 @@ + 'boolean' + ]; + /** + * @return \Illuminate\Database\Eloquent\Relations\HasMany + */ + public function ticketComments() + { + return $this->hasMany('TomatoPHP\FilamentSupport\Models\TicketComment'); + } + + /** + * @return \Illuminate\Database\Eloquent\Relations\BelongsTo + */ + public function type() + { + return $this->belongsTo('TomatoPHP\FilamentSupport\Models\Type'); + } + + public function accountable(){ + return $this->morphTo(); + } +} diff --git a/src/Models/TicketComment.php b/src/Models/TicketComment.php new file mode 100644 index 0000000..86eed6d --- /dev/null +++ b/src/Models/TicketComment.php @@ -0,0 +1,31 @@ +belongsTo('TomatoPHP\FilamentSupport\Models\Ticket'); + } +} diff --git a/testbench.yaml b/testbench.yaml new file mode 100644 index 0000000..fadafb2 --- /dev/null +++ b/testbench.yaml @@ -0,0 +1,24 @@ +providers: + - BladeUI\Icons\BladeIconsServiceProvider + - BladeUI\Heroicons\BladeHeroiconsServiceProvider + - Filament\Actions\ActionsServiceProvider + - Filament\FilamentServiceProvider + - Filament\Forms\FormsServiceProvider + - Filament\Infolists\InfolistsServiceProvider + - Filament\Notifications\NotificationsServiceProvider + - Filament\Support\SupportServiceProvider + - Filament\Tables\TablesServiceProvider + - Filament\Widgets\WidgetsServiceProvider + - RyanChandler\BladeCaptureDirective\BladeCaptureDirectiveServiceProvider + - TomatoPHP\FilamentSupport\FilamentSupportServiceProvider + - TomatoPHP\FilamentSupport\Tests\AdminPanelProvider +workbench: + welcome: true + install: true + start: / + guard: testing + discovers: + web: true + api: false + commands: false + views: true diff --git a/tests/Pest.php b/tests/Pest.php new file mode 100644 index 0000000..292996f --- /dev/null +++ b/tests/Pest.php @@ -0,0 +1,5 @@ +in(__DIR__); diff --git a/tests/database/database.sqlite b/tests/database/database.sqlite new file mode 100644 index 0000000000000000000000000000000000000000..cb9c4567851a96933c2d20eb6c1440eeb72754e6 GIT binary patch literal 245760 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 NswK-7XUpQz{{@MIBj5l4 literal 0 HcmV?d00001 diff --git a/tests/database/factories/.gitkeep b/tests/database/factories/.gitkeep new file mode 100644 index 0000000..e69de29 diff --git a/tests/database/factories/UserFactory.php b/tests/database/factories/UserFactory.php new file mode 100644 index 0000000..fd7db0f --- /dev/null +++ b/tests/database/factories/UserFactory.php @@ -0,0 +1,23 @@ + $this->faker->name(), + 'email' => $this->faker->unique()->safeEmail(), + 'email_verified_at' => now(), + 'password' => '$2y$10$92IXUNpkjO0rOQ5byMi.Ye4oKoEa3Ro9llC/.og/at2.uheWG/igi', // password + 'remember_token' => Str::random(10), + ]; + } +} diff --git a/tests/src/AdminPanelProvider.php b/tests/src/AdminPanelProvider.php new file mode 100644 index 0000000..388e873 --- /dev/null +++ b/tests/src/AdminPanelProvider.php @@ -0,0 +1,50 @@ +default() + ->id('admin') + ->path('admin') + ->login() + ->pages([ + Pages\Dashboard::class, + ]) + ->plugin( + FilamentSupportPlugin::make() + ) + ->middleware([ + EncryptCookies::class, + AddQueuedCookiesToResponse::class, + StartSession::class, + AuthenticateSession::class, + ShareErrorsFromSession::class, + VerifyCsrfToken::class, + SubstituteBindings::class, + DisableBladeIconComponents::class, + DispatchServingFilamentEvent::class, + ]) + ->authMiddleware([ + Authenticate::class, + ]); + } +} diff --git a/tests/src/DebugTest.php b/tests/src/DebugTest.php new file mode 100644 index 0000000..91bf6cd --- /dev/null +++ b/tests/src/DebugTest.php @@ -0,0 +1,5 @@ +each->not->toBeUsed(); +}); diff --git a/tests/src/Models/.gitkeep b/tests/src/Models/.gitkeep new file mode 100644 index 0000000..e69de29 diff --git a/tests/src/Models/User.php b/tests/src/Models/User.php new file mode 100644 index 0000000..43561e6 --- /dev/null +++ b/tests/src/Models/User.php @@ -0,0 +1,34 @@ +set('database.default', 'sqlite'); + $app['config']->set('database.connections.sqlite.database', __DIR__ . '/../database/database.sqlite'); + + $app['config']->set('view.paths', [ + ...$app['config']->get('view.paths'), + __DIR__ . '/../resources/views', + ]); + } +}