将不同语言的 post table 的 2 个相同 ID 存储到数据库中

Storing 2 same id of post table with different language into database

我正在建立一个多语言网站。在这种情况下只有 2 种语言,使用 Laravel 的印度尼西亚语和英语。 我有 posts table,它将存储每个 post 的 id 和 post_translations table 是存储 [=47= 的本地,标题和描述]. 将数据存储到数据库时出现问题。我不知道如何在不包含 ID 的情况下存储 post,除非我添加了 2 个相同的 post 和印度尼西亚语和英语。

这是结果(错误)

posts table

id 
1
2

post_translations table

id  post_id  locale   title
1     1      en       fisrt post
2     2      id       post yang pertama

扩展结果

posts table

id 
1

post_translations table

id  post_id  locale   title
1     1      en       fisrt post
2     1      id       post yang pertama

后控制器

  public function store(Request $request) {
    $this->validate($request, [
        'title' => 'required',
        'slug' => 'required',
        'content' => 'required'
    ]);

    $post = new Post;

    $post->title = $request->title;
    $post->slug = $request->slug;
    $post->content = $request->content;

    $post->save();
    return redirect()->route('post.index');
}

我认为你应该改变你的表结构:

posts : id, slug

post_translations : id, post_id, locale, title, content

同时添加与您的 Post 模型的关系:

public function translations()
{
    return $this->hasMany(PostTranslation::class, 'post_id');
}

并更新您的控制器:

$post = new Post;

$post->slug = $request->slug;
$post->save();

$post->translations()->create([
    'locale' => 'en', //or grab it from $request
    'title' => $request->title,
    'content' => $request->content
])

它将创建您的 post 并向其添加翻译

好的,我们开始吧(请注意,这不是唯一的方法):

  1. 安装 spatie/laravel-translatable 和

    作曲家要求spatie/laravel-translatable

**注意:对于本机 spatie/laravel-translatable 转到版本 2 **

  1. 创建一个具有以下结构的 table:

    创建 TABLE articles ( id int(10) UNSIGNED NOT NULL, title 文本整理 utf8_unicode_ci, slug 文本整理 utf8_unicode_ci, content 文本整理 utf8_unicode_ci, created_at 时间戳 NULL 默认 NULL, updated_at 时间戳 NULL 默认 NULL ) 引擎=InnoDB 默认字符集=utf8 整理=utf8_unicode_ci;

注意:最好使用迁移。我刚刚导出了一个 table 我之前做的测试

  1. 在数据库中以 json 格式插入数据,如下所示:

    INSERT INTO `pages` (`id`, `title`, `slug`, `content`, `created_at`, `updated_at`) VALUES
        (1, '{"ro":"Acas\u0103","en":"Home"}', NULL, '{"ro":"<p><strong>Test title<\/strong><\/p>\r\n\r\n<p>Test content romanian<\/p>\r\n","en":"<p><strong>test title english<\/strong><\/p>\r\n\r\n<p>test content english.<\/p>\r\n"}', '2017-04-03 11:45:56', '2017-04-03 12:15:16');

  1. 现在创建 blade 以编辑、更新、创建节目等。要获得语言,请在 blade 中执行类似的操作:

    {{ $data->title }}

    {!! nl2br($data->content) !!}

  2. 并且在控制器中: 添加如下内容:

    /** * 按语言生成字段 * * @param Model $entry 从数据库中选择的项目 * * @return 数组 */

    public function getTranslatableFields($fields)
    {
        $inputs = [];
        $languages = $this->getLanguages();
    
        foreach ($languages as $language) {
            foreach ($fields as $field) {
                $inputs[] = [
                    'name' => "{$field['name']}[{$language->abbr}]",
                    'label' => $field['label'] . " ($language->abbr)",
                    'lang' => $language->abbr,
                    'type' => array_key_exists('type', $field) ?     $field['type'] : 'text'
                ];
            }
        }
    
        return $inputs;
    }
    

我在 LangTrait 中添加了这个函数。因为我也用背包 laravel 我做了更多的事情。

为了编辑,我在特征中添加了这个方法:

/**
     * Show the form for editing the specified resource.
     *
     * @param   int             $id             the item's identifier
     *
     * @return  Response
     */
    public function edit($id)
    {    
        $data['entry'] = Model::find($id);
        $data['title'] = trans('lang_file.edit').' '.$this->entity_name; // name of the page 
        $data['fields'] = $this->getMultiLangFields($data['entry']);
        $data['id'] = $id;

        return view('crud::edit', $data);
    }


/**
     * Generate the field by language
     *
     * @param   Model        $entry      the item selected from the database
     *
     * @return  array
     */
    protected function getMultiLangFields($entry)
    {
        $fields['id'] = ['name' => 'id', 'type' => 'hidden', 'value' => $entry->id];

        foreach ($this->crud->update_fields as $key => $field) {
            $value = null;

            if (array_key_exists('lang', $field)) {
                $name = preg_replace('/(\[\w{2}\])$/i', '', $field['name']);
                $value = $entry->getTranslation($name, $field['lang']);
            }

            $fields[$key] = array_merge($field, ['value' => $value]);
        }

        return $fields;
    }



/**
     * Get the application active languages
     *
     * @return  \Backpack\LangFileManager\app\Models\Language
     */
    protected function getLanguages()
    {
        return Language::whereActive(1)->orderBy('default', 'desc')->get();
    }
  1. 最后在我的主控制器中我做了: 使用 LangTrait;(包含以上所有内容)

在构造中我添加了这个:

$this->getTranslatableFields($fields)

其中 $fields 是我需要的字段列表

所有方法都应该适应你的html格式。正如我所说,我将背包用于 laravel 并且相应地格式化字段

  1. 最后,为了让 getLanguage 文件生效,我在数据库中创建了一个新的 table 和一个模型:

型号:

class Language extends Model
{
    protected $table = 'languages';

    protected $fillable = ['name', 'flag', 'abbr', 'native', 'active', 'default'];

    public $timestamps = false;

    public static function getActiveLanguagesArray()
    {
        $active_languages = self::where('active', 1)->get()->toArray();
        $localizable_languages_array = [];

        if (count($active_languages)) {
            foreach ($active_languages as $key => $lang) {
                $localizable_languages_array[$lang['abbr']] = $lang;
            }

            return $localizable_languages_array;
        }

        return config('laravellocalization.supportedLocales');
    }

    public static function findByAbbr($abbr = false)
    {
        return self::where('abbr', $abbr)->first();
    }
}

Table:

CREATE TABLE `languages` (
  `id` int(10) UNSIGNED NOT NULL,
  `name` varchar(100) COLLATE utf8_unicode_ci NOT NULL,
  `app_name` varchar(100) COLLATE utf8_unicode_ci NOT NULL,
  `flag` varchar(100) COLLATE utf8_unicode_ci DEFAULT NULL,
  `abbr` varchar(3) COLLATE utf8_unicode_ci NOT NULL,
  `script` varchar(20) COLLATE utf8_unicode_ci DEFAULT NULL,
  `native` varchar(20) COLLATE utf8_unicode_ci DEFAULT NULL,
  `active` tinyint(3) UNSIGNED NOT NULL DEFAULT '1',
  `default` tinyint(3) UNSIGNED NOT NULL DEFAULT '0',
  `created_at` timestamp NULL DEFAULT NULL,
  `updated_at` timestamp NULL DEFAULT NULL,
  `deleted_at` timestamp NULL DEFAULT NULL
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci;

table中的数据:

INSERT INTO `languages` (`id`, `name`, `app_name`, `flag`, `abbr`, `script`, `native`, `active`, `default`, `created_at`, `updated_at`, `deleted_at`) VALUES
(1, 'English', 'english', '', 'en', 'Latn', 'English', 1, 0, NULL, NULL, NULL),
(2, 'Romanian', 'romanian', '', 'ro', 'Latn', 'română', 1, 1, NULL, NULL, NULL);

因为我是通过一个包来做的,所以我有点乱用代码。


现在,对于 spatie/laravel-translatable 包版本

  1. 在 config/app 中设置服务提供者。php 在提供者数组中添加:

    Spatie\Translatable\TranslatableServiceProvider::class,

在模型文章中添加 use HasTranslations; 像这样:

use Illuminate\Database\Eloquent\Model;
use Spatie\Translatable\HasTranslations;

class NewsItem extends Model
{
    use HasTranslations;

    public $translatable = ['name']; // list the columns you want to be translatable (will have json format)
}
  1. 保存一个新条目并使用它:

    $article = 新文章; $article->setTranslation('name', 'en', 'Updated name in English') ->setTranslation('name', 'nl', 'Naam in het Nederlands');

    $article->保存();

    $文章->姓名; // Returns 'Name in English' 鉴于当前应用程序区域设置为 'en' $article->getTranslation('name', 'nl'); // returns 'Naam in het Nederlands'

    app()->setLocale('nl');

    $文章->姓名; // Returns 'Naam in het Nederlands'

  2. 示例来自:https://github.com/spatie/laravel-translatable

  3. 数据库table格式如上第一版所述

如果不成功请告诉我,我会查看您的代码。