forked from qt-creator/qt-creator
		
	This option allows to define transactions for ValuesChangedCommand and ValuesModifiedCommand. Change-Id: Ia603027b5c431f1a1912d8959561f92a19fd0243 Reviewed-by: Miikka Heikkinen <miikka.heikkinen@qt.io>
		
			
				
	
	
		
			187 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			187 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/****************************************************************************
 | 
						|
**
 | 
						|
** Copyright (C) 2016 The Qt Company Ltd.
 | 
						|
** Contact: https://www.qt.io/licensing/
 | 
						|
**
 | 
						|
** This file is part of Qt Creator.
 | 
						|
**
 | 
						|
** Commercial License Usage
 | 
						|
** Licensees holding valid commercial Qt licenses may use this file in
 | 
						|
** accordance with the commercial license agreement provided with the
 | 
						|
** Software or, alternatively, in accordance with the terms contained in
 | 
						|
** a written agreement between you and The Qt Company. For licensing terms
 | 
						|
** and conditions see https://www.qt.io/terms-conditions. For further
 | 
						|
** information use the contact form at https://www.qt.io/contact-us.
 | 
						|
**
 | 
						|
** GNU General Public License Usage
 | 
						|
** Alternatively, this file may be used under the terms of the GNU
 | 
						|
** General Public License version 3 as published by the Free Software
 | 
						|
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
 | 
						|
** included in the packaging of this file. Please review the following
 | 
						|
** information to ensure the GNU General Public License requirements will
 | 
						|
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
 | 
						|
**
 | 
						|
****************************************************************************/
 | 
						|
 | 
						|
#include "valueschangedcommand.h"
 | 
						|
 | 
						|
#include "sharedmemory.h"
 | 
						|
#include <QCache>
 | 
						|
#include <QDebug>
 | 
						|
 | 
						|
#include <cstring>
 | 
						|
 | 
						|
#include <algorithm>
 | 
						|
 | 
						|
namespace QmlDesigner {
 | 
						|
 | 
						|
// using cache as a container which deletes sharedmemory pointers at process exit
 | 
						|
using GlobalSharedMemoryContainer = QCache<qint32, SharedMemory>;
 | 
						|
Q_GLOBAL_STATIC_WITH_ARGS(GlobalSharedMemoryContainer, globalSharedMemoryContainer, (10000))
 | 
						|
 | 
						|
ValuesChangedCommand::ValuesChangedCommand()
 | 
						|
    : m_keyNumber(0)
 | 
						|
{
 | 
						|
}
 | 
						|
 | 
						|
ValuesChangedCommand::ValuesChangedCommand(const QVector<PropertyValueContainer> &valueChangeVector)
 | 
						|
    : m_valueChangeVector (valueChangeVector),
 | 
						|
      m_keyNumber(0)
 | 
						|
{
 | 
						|
}
 | 
						|
 | 
						|
const QVector<PropertyValueContainer> ValuesChangedCommand::valueChanges() const
 | 
						|
{
 | 
						|
    return m_valueChangeVector;
 | 
						|
}
 | 
						|
 | 
						|
quint32 ValuesChangedCommand::keyNumber() const
 | 
						|
{
 | 
						|
    return m_keyNumber;
 | 
						|
}
 | 
						|
 | 
						|
void ValuesChangedCommand::removeSharedMemorys(const QVector<qint32> &keyNumberVector)
 | 
						|
{
 | 
						|
    for (qint32 keyNumber : keyNumberVector) {
 | 
						|
        SharedMemory *sharedMemory = globalSharedMemoryContainer()->take(keyNumber);
 | 
						|
        delete sharedMemory;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
void ValuesChangedCommand::sort()
 | 
						|
{
 | 
						|
    std::sort(m_valueChangeVector.begin(), m_valueChangeVector.end());
 | 
						|
}
 | 
						|
 | 
						|
static const QLatin1String valueKeyTemplateString("Values-%1");
 | 
						|
 | 
						|
static SharedMemory *createSharedMemory(qint32 key, int byteCount)
 | 
						|
{
 | 
						|
    SharedMemory *sharedMemory = new SharedMemory(QString(valueKeyTemplateString).arg(key));
 | 
						|
 | 
						|
    bool sharedMemoryIsCreated = sharedMemory->create(byteCount);
 | 
						|
 | 
						|
    if (sharedMemoryIsCreated) {
 | 
						|
        globalSharedMemoryContainer()->insert(key, sharedMemory);
 | 
						|
        return sharedMemory;
 | 
						|
    } else {
 | 
						|
        delete sharedMemory;
 | 
						|
    }
 | 
						|
 | 
						|
    return nullptr;
 | 
						|
}
 | 
						|
 | 
						|
QDataStream &operator<<(QDataStream &out, const ValuesChangedCommand &command)
 | 
						|
{
 | 
						|
    static const bool dontUseSharedMemory = qEnvironmentVariableIsSet("DESIGNER_DONT_USE_SHARED_MEMORY");
 | 
						|
 | 
						|
    QVector<PropertyValueContainer> propertyValueContainer = command.valueChanges();
 | 
						|
 | 
						|
    if (command.transactionOption != ValuesChangedCommand::TransactionOption::None) {
 | 
						|
        PropertyValueContainer optionContainer(command.transactionOption);
 | 
						|
        propertyValueContainer.append(optionContainer);
 | 
						|
    }
 | 
						|
 | 
						|
    if (!dontUseSharedMemory && propertyValueContainer.count() > 5) {
 | 
						|
        static quint32 keyCounter = 0;
 | 
						|
        ++keyCounter;
 | 
						|
        command.m_keyNumber = keyCounter;
 | 
						|
        QByteArray outDataStreamByteArray;
 | 
						|
        QDataStream temporaryOutDataStream(&outDataStreamByteArray, QIODevice::WriteOnly);
 | 
						|
        temporaryOutDataStream.setVersion(QDataStream::Qt_4_8);
 | 
						|
 | 
						|
        temporaryOutDataStream << propertyValueContainer;
 | 
						|
 | 
						|
        SharedMemory *sharedMemory = createSharedMemory(keyCounter, outDataStreamByteArray.size());
 | 
						|
 | 
						|
        if (sharedMemory) {
 | 
						|
            sharedMemory->lock();
 | 
						|
            std::memcpy(sharedMemory->data(), outDataStreamByteArray.constData(), sharedMemory->size());
 | 
						|
            sharedMemory->unlock();
 | 
						|
 | 
						|
            out << command.keyNumber();
 | 
						|
            return out;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    out << qint32(0);
 | 
						|
    out << propertyValueContainer;
 | 
						|
 | 
						|
    return out;
 | 
						|
}
 | 
						|
 | 
						|
void readSharedMemory(qint32 key, QVector<PropertyValueContainer> *valueChangeVector)
 | 
						|
{
 | 
						|
    SharedMemory sharedMemory(QString(valueKeyTemplateString).arg(key));
 | 
						|
    bool canAttach = sharedMemory.attach(QSharedMemory::ReadOnly);
 | 
						|
 | 
						|
    if (canAttach) {
 | 
						|
        sharedMemory.lock();
 | 
						|
 | 
						|
        QDataStream in(QByteArray::fromRawData(static_cast<const char*>(sharedMemory.constData()), sharedMemory.size()));
 | 
						|
        in.setVersion(QDataStream::Qt_4_8);
 | 
						|
        in >> *valueChangeVector;
 | 
						|
 | 
						|
        sharedMemory.unlock();
 | 
						|
        sharedMemory.detach();
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
QDataStream &operator>>(QDataStream &in, ValuesChangedCommand &command)
 | 
						|
{
 | 
						|
    in >> command.m_keyNumber;
 | 
						|
 | 
						|
    QVector<PropertyValueContainer> valueChangeVector;
 | 
						|
 | 
						|
    if (command.keyNumber() > 0)
 | 
						|
        readSharedMemory(command.keyNumber(), &valueChangeVector);
 | 
						|
    else
 | 
						|
        in >> valueChangeVector;
 | 
						|
 | 
						|
    // '-option-' is not a valid property name and indicates that we store the transaction option.
 | 
						|
    if (!valueChangeVector.isEmpty() && valueChangeVector.last().name() == "-option-") {
 | 
						|
        command.transactionOption =
 | 
						|
            static_cast<ValuesChangedCommand::TransactionOption>(valueChangeVector.last().instanceId());
 | 
						|
        valueChangeVector.removeLast();
 | 
						|
    }
 | 
						|
 | 
						|
    command.m_valueChangeVector = valueChangeVector;
 | 
						|
 | 
						|
    return in;
 | 
						|
}
 | 
						|
 | 
						|
bool operator ==(const ValuesChangedCommand &first, const ValuesChangedCommand &second)
 | 
						|
{
 | 
						|
    return first.m_valueChangeVector == second.m_valueChangeVector
 | 
						|
        && first.transactionOption == second.transactionOption;
 | 
						|
}
 | 
						|
 | 
						|
QDebug operator <<(QDebug debug, const ValuesChangedCommand &command)
 | 
						|
{
 | 
						|
    return debug.nospace() << "ValuesChangedCommand("
 | 
						|
                    << "keyNumber: " << command.keyNumber() << ", "
 | 
						|
                    << command.valueChanges() << ")";
 | 
						|
}
 | 
						|
 | 
						|
} // namespace QmlDesigner
 |