”工欲善其事,必先利其器。“—孔子《论语.录灵公》
首页 > 编程 > 使用自动加载翻译在 Laravel 中构建多态可翻译模型

使用自动加载翻译在 Laravel 中构建多态可翻译模型

发布于2024-08-18
浏览:243

Building a Polymorphic Translatable Model in Laravel with Autoloaded Translations

处理多语言内容时,将翻译存储在 JSON 列中通常比每个属性的单独行存储更有效。这种方法将翻译整合到单个列中,从而简化了数据管理和检索。

设置翻译系统

我们将增强翻译模型和表,以使用 JSON 列来存储翻译。这将涉及更新表架构并修改 Translatable 特征以处理 JSON 数据。

第 1 步:创建翻译表迁移

如果翻译表尚不存在,则创建一个新的迁移:

php artisan make:migration create_translations_table

第 2 步:定义表结构

在database/migrations中打开生成的迁移文件。对于新表,定义如下:

use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;

class CreateTranslationsTable extends Migration
{
    public function up()
    {
        Schema::create('translations', function (Blueprint $table) {
            $table->id();
            $table->string('locale'); // Stores the locale, e.g., 'en', 'fr'
            $table->string('translatable_type'); // Stores the related model type, e.g., 'Post', 'Product'
            $table->unsignedBigInteger('translatable_id'); // Stores the ID of the related model
            $table->json('translations'); // Stores all translations as a JSON object
            $table->timestamps();
        });
    }

    public function down()
    {
        Schema::dropIfExists('translations');
    }
}

第 3 步:运行迁移
将迁移应用到您的数据库:

php artisan migrate

第 4 步:创建翻译模型

接下来,创建翻译模型来处理多态关系:

php artisan make:model Translation

Translation模型中,定义多态关系:

class Translation extends Model
{
    protected $fillable = ['locale', 'translatable_type', 'translatable_id', 'translations'];

    protected $casts = [
        'translations' => 'array',
    ];

    public function translatable()
    {
        return $this->morphTo();
    }
}

使用 JSON 存储实现可翻译特征

为了使翻译处理可在多个模型中重用,我们将创建一个 Translatable 特征,它将根据用户选择的区域设置自动加载翻译内容。此外,如果所选语言环境没有可用的翻译,我们将添加后备机制以从默认语言环境加载内容。

第 1 步:使用 JSON 处理创建可翻译特征

namespace App\Traits;

use App\Models\Translation;
use Illuminate\Support\Facades\App;

trait Translatable
{
    public static function bootTranslatable()
    {
        static::retrieved(function ($model) {
            $model->loadTranslations();
        });
    }

    public function translations()
    {
        return $this->morphMany(Translation::class, 'translatable');
    }

    public function loadTranslations()
    {
        $locale = App::getLocale();
        $defaultLocale = config('app.default_locale', 'en'); // Fallback to the default locale

        // Try to load translations for the current locale
        $translation = $this->translations()->where('locale', $locale)->first();

        if (!$translation && $locale !== $defaultLocale) {
            // If no translations are found for the current locale, fallback to the default locale
            $translation = $this->translations()->where('locale', $defaultLocale)->first();
        }

        if ($translation) {
            $translations = $translation->translations;
            foreach ($translations as $key => $value) {
                $this->{$key} = $value;
            }
        }
    }

    public function addTranslations(array $translations, $locale = null)
    {
        $locale = $locale ?? App::getLocale();
        return $this->translations()->updateOrCreate(
            ['locale' => $locale],
            ['translations' => $translations]
        );
    }
}

第 2 步:将可翻译特征应用于您的模型
将 Translatable 特征添加到任何需要翻译支持的模型中。

namespace App\Models;

use App\Traits\Translatable;
use Illuminate\Database\Eloquent\Model;

class Post extends Model
{
    use Translatable;

    protected $fillable = ['title', 'content'];
}

示例:创建翻译模型

将翻译添加为 JSON 对象:

$post = Post::create(['title' => 'Default Title', 'content' => 'Default Content']);

// Adding translations
$post->addTranslations([
    'title' => 'Hello World',
    'content' => 'Welcome to our website'
], 'en');

$post->addTranslations([
    'title' => 'Bonjour le monde',
    'content' => 'Bienvenue sur notre site Web'
], 'fr');

检索翻译模型

当您检索 Post 模型时,它将根据当前语言环境自动加载翻译内容,或者在必要时回退到默认语言环境:

App::setLocale('fr');
$post = Post::find(1);
echo $post->title; // Displays "Bonjour le monde" if French translation exists

App::setLocale('es');
$post = Post::find(1);
echo $post->title; // Displays "Hello World" as it falls back to the English translation

在视图中显示翻译内容

在 Blade 视图中,您可以像任何其他模型属性一样显示翻译的内容:

{{ $post->title }}

{{ $post->content }}

结论

通过使用 JSON 列来存储翻译并实现回退机制,您可以简化 Laravel 应用程序中多语言内容的管理。这种方法将翻译整合到单个列中,简化了数据处理并使您的代码库更易于维护。无论您是构建博客、电子商务网站还是任何多语言应用程序,此方法都能确保流畅高效的用户体验。

享受!

版本声明 本文转载于:https://dev.to/rafaelogic/building-a-polymorphic-translatable-model-in-laravel-with-autoloaded-translations-3d99?1如有侵犯,请联系[email protected]删除
最新教程 更多>
  • 寻找经济实惠的同日格兰尼公寓(带 Pillar Build Granny Flats)
    寻找经济实惠的同日格兰尼公寓(带 Pillar Build Granny Flats)
    在 Pillar Build Granny Flats,我们为您提供祖母屋解决方案的精英服务,满足您的独特需求。无论是房主、承包商还是投资者,我们都可以帮助您在当天购买后院公寓,效果非常好,为您节省宝贵的时间,而且不用说,预算也很实惠。我们的祖母房建造者将在每一步工作,以确保您的项目以最精确和细心的...
    编程 发布于2024-11-05
  • 如何使用 botoith Google Colab 和 AWS 集成
    如何使用 botoith Google Colab 和 AWS 集成
    您有没有想过,在实施AWS Lambda时,想要一一确认代码的运行情况? 您可能认为在 AWS 控制台上实施很痛苦,因为您必须运行 Lambda 函数并且每次都会产生成本。 因此,我将向您展示您的担忧的解决方案。 它是通过 Google Colab 和 AWS 集成实现的。 步骤如下: ...
    编程 发布于2024-11-05
  • (高性能 Web 应用程序的要求
    (高性能 Web 应用程序的要求
    “高性能网络应用程序”或“前端”到底是什么? 自从 Internet Explorer 时代衰落以来,JavaScript 生态系统变得越来越强大,“前端”一词已成为高性能、现代 Web 客户端的代名词。这个“前端”世界的核心是 React。事实上,在前端开发中不使用 React 常常会让一个人看...
    编程 发布于2024-11-05
  • 如何将单个输入字段设置为分区输入?
    如何将单个输入字段设置为分区输入?
    将输入字段设置为分区输入有多种方法可用于创建一系列分区输入字段。一种方法利用“字母间距”来分隔单个输入字段内的字符。此外,“background-image”和“border-bottom”样式可以进一步增强多个输入字段的错觉。CSS Snippet以下 CSS 代码演示了如何创建所需的效果:#pa...
    编程 发布于2024-11-05
  • 用 Go 构建一个简单的负载均衡器
    用 Go 构建一个简单的负载均衡器
    负载均衡器在现代软件开发中至关重要。如果您曾经想知道如何在多个服务器之间分配请求,或者为什么某些网站即使在流量大的情况下也感觉更快,答案通常在于高效的负载平衡。 在这篇文章中,我们将使用 Go 中的循环算法构建一个简单的应用程序负载均衡器。这篇文章的目的是逐步了解负载均衡器的工作原理。 ...
    编程 发布于2024-11-05
  • 如何以超链接方式打开本地目录?
    如何以超链接方式打开本地目录?
    通过超链接导航本地目录尝试在链接交互时启动本地目录视图时,您可能会遇到限制。然而,有一个解决方案可以解决这个问题,并且可以在各种浏览器之间无缝工作。实现方法因为从 HTML 页面直接打开路径或启动浏览器是由于安全原因受到限制,更可行的方法是提供可下载的链接(.URL 或 .LNK)。推荐路径:.UR...
    编程 发布于2024-11-05
  • 为什么 Makefile 会抛出 Go 命令的权限被拒绝错误?
    为什么 Makefile 会抛出 Go 命令的权限被拒绝错误?
    运行 Go 时 Makefile 中出现权限被拒绝错误通过 Makefile 运行 Go 命令时可能会遇到“权限被拒绝”错误,即使你可以直接执行它们。这种差异是由于 GNU make 中的问题引起的。原因:当您的 PATH 上有一个目录包含名为“go.gnu”的子目录时,就会出现此错误。 ”例如,如...
    编程 发布于2024-11-05
  • parseInt 函数中 Radix 参数的意义是什么?
    parseInt 函数中 Radix 参数的意义是什么?
    parseInt 函数中 Radix 的作用parseInt 函数将字符串转换为整数。然而,它并不总是采用以 10 为基数的数字系统。要指定所需的基数,请使用基数参数。理解基数基数是指单个数字表示的值的数量。例如,十六进制的基数为 16,八进制的基数为 8,二进制的基数为 2。为什么使用基数?需要当...
    编程 发布于2024-11-05
  • 在空数据集上使用 MySQL 的 SUM 函数时如何返回“0”而不是 NULL?
    在空数据集上使用 MySQL 的 SUM 函数时如何返回“0”而不是 NULL?
    当不存在任何值时如何从 MySQL 的 SUM 函数中检索“0”MySQL 中的 SUM 函数提供了一种方便的方法来聚合数值价值观。但是,当查询期间没有找到匹配的行时,SUM 函数通常返回 NULL 值。对于某些用例,可能更需要返回“0”而不是 NULL。利用 COALESCE 解决问题此问题的解决...
    编程 发布于2024-11-05
  • 如何使用 JavaScript 将链接保留在同一选项卡中?
    如何使用 JavaScript 将链接保留在同一选项卡中?
    在同一选项卡和窗口中导航链接您可能会遇到想要在同一窗口和选项卡中打开链接的情况作为当前页面。但是,使用 window.open 函数通常会导致在新选项卡中打开链接。为了解决这个问题,您可以使用 name 属性,如下所示:window.open("https://www.youraddres...
    编程 发布于2024-11-05
  • 如何解决Python中的循环依赖?
    如何解决Python中的循环依赖?
    Python 中的循环依赖使用 Python 模块时遇到循环依赖可能是一个令人沮丧的问题。在这个特定场景中,我们有两个文件,node.py 和 path.py,分别包含 Node 和 Path 类。最初,path.py 使用 from node.py import * 导入 node.py。但是,在...
    编程 发布于2024-11-05
  • MariaDB 与 MySQL:开发人员需要了解什么
    MariaDB 与 MySQL:开发人员需要了解什么
    MariaDB 和 MySQL 是著名的开源 RDBMS,但尽管它们有着共同的历史,但它们在功能和性能方面却有所不同。本文快速强调了主要差异,帮助开发人员决定哪个数据库最适合他们的需求。 差异和示例 存储引擎,MariaDB 对 Aria 和 MyRocks 等引擎的扩展支持提供了比...
    编程 发布于2024-11-05
  • 为什么我的 Goroutine 递增变量会产生意外的结果?
    为什么我的 Goroutine 递增变量会产生意外的结果?
    这是编译器优化的结果吗?在此代码片段中,启动了一个 goroutine 并重复递增变量 i:package main import "time" func main() { i := 1 go func() { for { ...
    编程 发布于2024-11-05
  • 利用 AI 快速学习 Node.js - 第 4 天
    利用 AI 快速学习 Node.js - 第 4 天
    今天,借助ChatGPT继续学习Node.js,重点是异步编程。这是 Node.js 中最重要的概念之一,我很高兴能够开始掌握它。 理论 在 Node.js 中,异步编程因其非阻塞、事件驱动的架构而至关重要。这意味着文件读取、数据库查询或网络请求等操作在等待结果时不会阻塞其他代码的执行。 我们探索了...
    编程 发布于2024-11-05
  • Java 可以定义带有嵌入引号的字符串而不转义吗?
    Java 可以定义带有嵌入引号的字符串而不转义吗?
    揭开 Java 使用嵌入式引号定义字符串的替代方法在 Java 中处理字符串时,您常常会在文字中遇到大量引号,导致繁琐的转义和可读性挑战。虽然其他语言提供了处理这种情况的语法,但 Java 缺乏类似的选项。问题: Java 是否提供了另一种方法来定义带有嵌入引号的字符串而不诉诸转义?答案: 虽然 J...
    编程 发布于2024-11-05

免责声明: 提供的所有资源部分来自互联网,如果有侵犯您的版权或其他权益,请说明详细缘由并提供版权或权益证明然后发到邮箱:[email protected] 我们会第一时间内为您处理。

Copyright© 2022 湘ICP备2022001581号-3