<?php
/**
 * @link http://www.yiiframework.com/
 * @copyright Copyright (c) 2008 Yii Software LLC
 * @license http://www.yiiframework.com/license/
 */

namespace yiiunit\framework\data;

use yii\data\ActiveDataProvider;
use yii\db\Query;
use yiiunit\data\ar\ActiveRecord;
use yiiunit\data\ar\Customer;
use yiiunit\data\ar\Item;
use yiiunit\framework\db\DatabaseTestCase;
use yiiunit\data\ar\Order;

/**
 * @author Qiang Xue <qiang.xue@gmail.com>
 * @since 2.0
 *
 * @group data
 * @group db
 */
class ActiveDataProviderTest extends DatabaseTestCase
{
    protected function setUp()
    {
        parent::setUp();
        ActiveRecord::$db = $this->getConnection();
    }

    public function testActiveQuery()
    {
        $provider = new ActiveDataProvider([
            'query' => Order::find()->orderBy('id'),
        ]);
        $orders = $provider->getModels();
        $this->assertEquals(3, count($orders));
        $this->assertTrue($orders[0] instanceof Order);
        $this->assertTrue($orders[1] instanceof Order);
        $this->assertTrue($orders[2] instanceof Order);
        $this->assertEquals([1, 2, 3], $provider->getKeys());

        $provider = new ActiveDataProvider([
            'query' => Order::find(),
            'pagination' => [
                'pageSize' => 2,
            ]
        ]);
        $orders = $provider->getModels();
        $this->assertEquals(2, count($orders));
    }

    public function testActiveRelation()
    {
        /** @var Customer $customer */
        $customer = Customer::findOne(2);
        $provider = new ActiveDataProvider([
            'query' => $customer->getOrders(),
        ]);
        $orders = $provider->getModels();
        $this->assertEquals(2, count($orders));
        $this->assertTrue($orders[0] instanceof Order);
        $this->assertTrue($orders[1] instanceof Order);
        $this->assertEquals([2, 3], $provider->getKeys());

        $provider = new ActiveDataProvider([
            'query' => $customer->getOrders(),
            'pagination' => [
                'pageSize' => 1,
            ]
        ]);
        $orders = $provider->getModels();
        $this->assertEquals(1, count($orders));
    }

    public function testActiveRelationVia()
    {
        /** @var Order $order */
        $order = Order::findOne(2);
        $provider = new ActiveDataProvider([
            'query' => $order->getItems(),
        ]);
        $items = $provider->getModels();
        $this->assertEquals(3, count($items));
        $this->assertTrue($items[0] instanceof Item);
        $this->assertTrue($items[1] instanceof Item);
        $this->assertTrue($items[2] instanceof Item);
        $this->assertEquals([3, 4, 5], $provider->getKeys());

        $provider = new ActiveDataProvider([
            'query' => $order->getItems(),
            'pagination' => [
                'pageSize' => 2,
            ]
        ]);
        $items = $provider->getModels();
        $this->assertEquals(2, count($items));
    }

    public function testActiveRelationViaTable()
    {
        /** @var Order $order */
        $order = Order::findOne(1);
        $provider = new ActiveDataProvider([
            'query' => $order->getBooks(),
        ]);
        $items = $provider->getModels();
        $this->assertEquals(2, count($items));
        $this->assertTrue($items[0] instanceof Item);
        $this->assertTrue($items[1] instanceof Item);

        $provider = new ActiveDataProvider([
            'query' => $order->getBooks(),
            'pagination' => [
                'pageSize' => 1,
            ]
        ]);
        $items = $provider->getModels();
        $this->assertEquals(1, count($items));
    }

    public function testQuery()
    {
        $query = new Query;
        $provider = new ActiveDataProvider([
            'db' => $this->getConnection(),
            'query' => $query->from('order')->orderBy('id'),
        ]);
        $orders = $provider->getModels();
        $this->assertEquals(3, count($orders));
        $this->assertTrue(is_array($orders[0]));
        $this->assertEquals([0, 1, 2], $provider->getKeys());

        $query = new Query;
        $provider = new ActiveDataProvider([
            'db' => $this->getConnection(),
            'query' => $query->from('order'),
            'pagination' => [
                'pageSize' => 2,
            ]
        ]);
        $orders = $provider->getModels();
        $this->assertEquals(2, count($orders));
    }

    public function testRefresh()
    {
        $query = new Query;
        $provider = new ActiveDataProvider([
            'db' => $this->getConnection(),
            'query' => $query->from('order')->orderBy('id'),
        ]);
        $this->assertEquals(3, count($provider->getModels()));

        $provider->getPagination()->pageSize = 2;
        $this->assertEquals(3, count($provider->getModels()));
        $provider->refresh();
        $this->assertEquals(2, count($provider->getModels()));
    }

    public function testPaginationBeforeModels()
    {
        $query = new Query;
        $provider = new ActiveDataProvider([
            'db' => $this->getConnection(),
            'query' => $query->from('order')->orderBy('id'),
        ]);
        $pagination = $provider->getPagination();
        $this->assertEquals(0, $pagination->getPageCount());
        $this->assertCount(3, $provider->getModels());
        $this->assertEquals(1, $pagination->getPageCount());

        $provider->getPagination()->pageSize = 2;
        $this->assertEquals(3, count($provider->getModels()));
        $provider->refresh();
        $this->assertEquals(2, count($provider->getModels()));
    }
}