Qt控件编辑功能(二)

简述

根据QtDesigner的控件选中,拉伸效果,用过Qt的盆友都很熟悉Qt的Designer,这个我就不多说了,我们先看看QtDesigner中的效果

Qt Designer效果图图

这里写图片描述

模仿功能介绍

1.支持选中效果;
2.支持自由拉伸效果;
3.支持双击鼠标左键编辑功能;
4.支持键盘 ↑↓←→按键移动;
5.支持按住ctrl + 鼠标左键 多选控件功能;
6.支持键盘delete键,删除选中控件功能;

模仿效果图

这里写图片描述
这里写图片描述
这里写图片描述

代码

//用法试例
formWindowManager = new FromWindowManager(this);
formWindowManager->addControlWidget(ui.label);
formWindowManager->addControlWidget(ui.pushButton);
formWindowManager->addControlWidget(ui.toolButton);
//窗口控件管理类
CompontEditor* FromWindowManagerPrivate::getWidgetForEditor(QWidget* widget)
{
    for (auto iter : editorlst){
        if (widget == iter->getEditorWidget()){
            return iter;
        }
    }
    return nullptr;
}

FromWindowManager::FromWindowManager(QWidget *parent)
    : QObject(parent)
    , d_ptr(new FromWindowManagerPrivate(this))
{
    parent->setFocus(Qt::MouseFocusReason);
    parent->installEventFilter(this);
}

FromWindowManager::~FromWindowManager()
{

}

void FromWindowManager::addControlWidget(QWidget* widget)
{
    Q_D(FromWindowManager);
    widget->installEventFilter(this);
    CompontEditor* editor = new CompontEditor(widget, widget->property("text").toString(), widget->parentWidget());
    d->editorlst << editor;
}

void FromWindowManager::setSelectWidget(CompontEditor* editorWidget)
{
    Q_D(FromWindowManager);
    d->selecteditorlst << editorWidget;
}

void FromWindowManager::clearSelectWidget(CompontEditor* editorWidget)
{
    Q_D(FromWindowManager);
    d->selecteditorlst.removeAll(editorWidget);
}

void FromWindowManager::removeControlWidget(CompontEditor* editorWidget)
{
    Q_D(FromWindowManager);
    editorWidget->getEditorWidget()->removeEventFilter(this);
    d->editorlst.removeAll(editorWidget);
}

bool FromWindowManager::eventFilter(QObject *watched, QEvent *event)
{
    switch (event->type()){
    case QEvent::KeyPress:{
        QKeyEvent* keyevent = dynamic_cast<QKeyEvent*>(event);
        Q_D(FromWindowManager);
        auto tmpselecteditorlst = d->selecteditorlst;
        for (auto iter : tmpselecteditorlst){
            QWidget* selectWidget = iter->getSelectWidget();
            if (keyevent->key() == Qt::Key_Left){
                selectWidget->move(selectWidget->pos() - QPoint(1, 0));
            }
            else if (keyevent->key() == Qt::Key_Right){
                selectWidget->move(selectWidget->pos() + QPoint(1, 0));
            }
            else if (keyevent->key() == Qt::Key_Up){
                selectWidget->move(selectWidget->pos() - QPoint(0, 1));
            }
            else if (keyevent->key() == Qt::Key_Down){
                selectWidget->move(selectWidget->pos() + QPoint(0, 1));
            }
            else if (keyevent->key() == Qt::Key_Delete){
                clearSelectWidget(iter);
                removeControlWidget(iter);
                delete iter->getEditorWidget();
                delete iter;                
            }
        }
        return true;
    }
        break;
    case QEvent::MouseButtonPress:{
        Q_D(FromWindowManager);
        auto editor =d->getWidgetForEditor(dynamic_cast<QWidget*>(watched));
        if (editor){
            if (editor->isSelect()){
                return false;
            }
            if (QApplication::keyboardModifiers() == Qt::ControlModifier){
                QMouseEvent* mouseevent = dynamic_cast<QMouseEvent*>(event);
                if (mouseevent->button() == Qt::LeftButton){
                    editor->setSelect();
                }
            }
            else{
                for (auto iter : d->selecteditorlst){
                    iter->clearSelect();
                }
                editor->setSelect();
                d->selecteditorlst.clear();
            }
            d->selecteditorlst << editor;
        }
        else {
            auto posWidget = qApp->widgetAt(QCursor::pos());
            Q_D(FromWindowManager);
            for (auto iter : d->editorlst){
                if (iter->getEditorWidget() == posWidget || 
                    iter->getSelectWidget() == posWidget ||
                    iter == posWidget){
                    return false;
                }
            }
            for (auto iter : d->selecteditorlst){
                iter->clearSelect();
            }
            d->selecteditorlst.clear();
        }
    }
         break;
    default:
        break;
    }
    return QObject::eventFilter(watched, event);
}
//选中控件类
SelectWidget::SelectWidget(QWidget *parent)
    : QWidget(parent)
    , d_ptr(new SelectWidgetPrivate(this))
{
    this->setMouseTracking(true);
    setObjectName("q_selectwidget");
}

SelectWidget::~SelectWidget()
{
    
}

void SelectWidget::paintEvent(QPaintEvent *event)
{
    QWidget::paintEvent(event);
    QPainter painter(this);
    painerPathRect(&painter);
}

void SelectWidget::painerPathRect(QPainter* painter)
{
    painter->setPen(Qt::NoPen);
    painter->setBrush(Qt::blue);
    painter->drawRect(0, 0, BACKRECT, BACKRECT);
    painter->drawRect(width() / 2 - BACKRECT / 2, 0, BACKRECT, BACKRECT);
    painter->drawRect(width() - BACKRECT, 0, BACKRECT, BACKRECT);
    painter->drawRect(0, height() / 2 - BACKRECT / 2, BACKRECT, BACKRECT);
    painter->drawRect(width() - BACKRECT, height() / 2 - BACKRECT / 2, BACKRECT, BACKRECT);
    painter->drawRect(0, height() - BACKRECT, BACKRECT, BACKRECT);
    painter->drawRect(width() / 2 - BACKRECT / 2, height() - BACKRECT, BACKRECT, BACKRECT);
    painter->drawRect(width() - BACKRECT, height() - BACKRECT, BACKRECT, BACKRECT);
}

void SelectWidget::mousePressEvent(QMouseEvent *event)
{
    Q_D(SelectWidget);
    if (event->button() == Qt::LeftButton){
        d->m_leftButtonPress = true;
    }
    d->m_mousepressPos = QCursor::pos();
    QWidget::mousePressEvent(event);
}

void SelectWidget::mouseMoveEvent(QMouseEvent *event)
{
    Q_D(SelectWidget);
    if (d->m_leftButtonPress && d->m_type == SelectWidgetPrivate::NORMAL){
        QPoint movepoint = QCursor::pos() - d->m_mousepressPos;
        d->m_mousepressPos = QCursor::pos();
        this->move(this->pos() + movepoint);
        d->m_uiwidget->setGeometry(QRect(this->pos() + QPoint(2, 2), this->size() + QSize(-4, -4)));
    }
    else if (d->m_leftButtonPress && d->m_type != SelectWidgetPrivate::NORMAL){
        resizeSection();
    }
    else{
        updateCursorType();
    }
    QWidget::mouseMoveEvent(event);
}

void SelectWidget::mouseReleaseEvent(QMouseEvent *event)
{
    Q_D(SelectWidget);
    d->m_mousepressPos = event->globalPos();
    d->m_leftButtonPress = false;
    QWidget::mouseReleaseEvent(event);
}

void SelectWidget::updateCursor()
{
    Q_D(SelectWidget);
    switch (d->m_type) {
    case SelectWidgetPrivate::LeftTop:
        this->setCursor(Qt::SizeFDiagCursor);
        break;
    case SelectWidgetPrivate::Top:
        this->setCursor(Qt::SizeVerCursor);
        break;
    case SelectWidgetPrivate::RightTop:
        this->setCursor(Qt::SizeBDiagCursor);
        break;
    case SelectWidgetPrivate::Right:
        this->setCursor(Qt::SizeHorCursor);
        break;
    case SelectWidgetPrivate::RightBottom:
        this->setCursor(Qt::SizeFDiagCursor);
        break;
    case SelectWidgetPrivate::Bottom:
        this->setCursor(Qt::SizeVerCursor);
        break;
    case SelectWidgetPrivate::LeftBottom:
        this->setCursor(Qt::SizeBDiagCursor);
        break;
    case SelectWidgetPrivate::Left:
        this->setCursor(Qt::SizeHorCursor);
        break;
    default:
        this->setCursor(Qt::ArrowCursor);
        break;
    }
}

void SelectWidget::resizeSection()
{
    Q_D(SelectWidget);
    QPoint mousemovePos = QCursor::pos();
    QPoint widgetGloabPoint(this->mapToGlobal(QPoint(0, 0)));
    QPoint widgetpoint = this->mapToParent(this->mapFromGlobal(mousemovePos));
    switch (d->m_type) {
    case SelectWidgetPrivate::LeftTop:{
        int resizeH = widgetGloabPoint.y() - mousemovePos.y() + this->height();
        int resizeW = widgetGloabPoint.x() - mousemovePos.x() + this->width();
        if (this->minimumHeight() <= resizeH && resizeH <= this->maximumHeight()){
            this->setGeometry(this->x(), widgetpoint.y(), resizeW, resizeH);
        }
        if (this->minimumWidth() <= resizeW && resizeW <= this->maximumWidth()){
            this->setGeometry(widgetpoint.x(), this->y(), resizeW, resizeH);
        }
    }
        break;
    case SelectWidgetPrivate::Top:{
        int resizeH = widgetGloabPoint.y() - mousemovePos.y() + this->height();
        if (this->minimumHeight() <= resizeH && resizeH <= this->maximumHeight()){
            this->setGeometry(this->x(), widgetpoint.y(), this->width(), resizeH);
        }
    }
        break;
    case SelectWidgetPrivate::RightTop:{
        int resizeH = widgetGloabPoint.y() + this->height() - mousemovePos.y();
        int resizeW = mousemovePos.x() - widgetGloabPoint.x();
        int pointY = widgetpoint.y();
        if (this->minimumHeight() <= resizeH && resizeH <= this->maximumHeight()){
            this->setGeometry(this->x(), widgetpoint.y(), resizeW, resizeH);
        }
    }
        break;
    case SelectWidgetPrivate::Right:{
        int resizeW = mousemovePos.x() - widgetGloabPoint.x();
        this->setGeometry(this->x(), this->y(), resizeW, this->height());
    }
        break;
    case SelectWidgetPrivate::RightBottom:{
        int resizeW = mousemovePos.x() - widgetGloabPoint.x();
        int resizeH = mousemovePos.y() - widgetGloabPoint.y();
        this->setGeometry(this->x(), this->y(), resizeW, resizeH);
    }
        break;
    case SelectWidgetPrivate::Bottom:{
        int resizeH = mousemovePos.y() - widgetGloabPoint.y();
        this->setGeometry(this->x(), this->y(), this->width(), resizeH);
    }
        break;
    case SelectWidgetPrivate::LeftBottom:{
        int resizeH = mousemovePos.y() - widgetGloabPoint.y();
        int resizeW = widgetGloabPoint.x() - mousemovePos.x() + this->width();
        if (this->minimumWidth() <= resizeW && resizeW <= this->maximumWidth()){
            this->setGeometry(widgetpoint.x(), this->y(), resizeW, resizeH);
        }
        if (this->minimumHeight() <= resizeH && resizeH <= this->maximumHeight()){
            this->setGeometry(this->x(), this->y(), resizeW, resizeH);
        }
    }
        break;
    case SelectWidgetPrivate::Left:{
        int resizeW = widgetGloabPoint.x() - mousemovePos.x() + this->width();
        if (this->minimumWidth() <= resizeW && resizeW <= this->maximumWidth()){
            this->setGeometry(widgetpoint.x(), this->y(), resizeW, this->height());
        }
    }
        break;
    default:{

    }
        break;
    }
    d->m_uiwidget->setGeometry(QRect(this->pos() + QPoint(2, 2), this->size() + QSize(-4, -4)));
}

void SelectWidget::moveEvent(QMoveEvent *event)
{
    Q_D(SelectWidget);
    d->m_uiwidget->move(this->pos() + QPoint(2, 2));
    QWidget::moveEvent(event);
}

void SelectWidget::updateCursorType()
{
    Q_D(SelectWidget);
    QPoint mousemovePos = QCursor::pos();
    QRect widgetGloabRect(this->mapToGlobal(QPoint(0, 0)), this->size());
    if (QRect(widgetGloabRect.bottomLeft() - QPoint(DISTANCE / 2, DISTANCE / 2), widgetGloabRect.bottomLeft() + QPoint(DISTANCE / 2, DISTANCE / 2)).contains(mousemovePos)){
        d->m_type = SelectWidgetPrivate::LeftBottom;
    }
    else if (QRect(widgetGloabRect.bottomRight() - QPoint(DISTANCE / 2, DISTANCE / 2), widgetGloabRect.bottomRight() + QPoint(DISTANCE / 2, DISTANCE / 2)).contains(mousemovePos)){
        d->m_type = SelectWidgetPrivate::RightBottom;
    }
    else if (QRect(widgetGloabRect.topRight() - QPoint(DISTANCE / 2, DISTANCE / 2), widgetGloabRect.topRight() + QPoint(DISTANCE / 2, DISTANCE / 2)).contains(mousemovePos)){
        d->m_type = SelectWidgetPrivate::RightTop;
    }
    else if (QRect(widgetGloabRect.topLeft() - QPoint(DISTANCE / 2, DISTANCE / 2), widgetGloabRect.topLeft() + QPoint(DISTANCE / 2, DISTANCE / 2)).contains(mousemovePos)){
        d->m_type = SelectWidgetPrivate::LeftTop;
    }
    else if (qAbs(mousemovePos.x() - widgetGloabRect.left()) < DISTANCE){
        d->m_type = SelectWidgetPrivate::Left;
    }
    else if (qAbs(mousemovePos.y() - widgetGloabRect.bottom()) < DISTANCE){
        d->m_type = SelectWidgetPrivate::Bottom;
    }
    else if (qAbs(mousemovePos.x() - widgetGloabRect.right()) < DISTANCE){
        d->m_type = SelectWidgetPrivate::Right;
    }
    else if (qAbs(mousemovePos.y() - widgetGloabRect.top()) < DISTANCE){
        d->m_type = SelectWidgetPrivate::Top;
    }
    else{
        d->m_type = SelectWidgetPrivate::NORMAL;
    }
    updateCursor();
}

void SelectWidget::setSelectWidget(QWidget* widget)
{
    Q_D(SelectWidget);
    d->m_uiwidget = widget;
}

QWidget* SelectWidget::getSelectWidget()
{
    Q_D(SelectWidget);
    return d->m_uiwidget;
}

void SelectWidget::setSelectWidgetMask(const QRect& rect)
{
    setMask(QRegion(rect));
}

void SelectWidget::resizeEvent(QResizeEvent *event)
{
    QPixmap pixmap(this->size());
    pixmap.fill(Qt::transparent);
    QPainter painter(&pixmap);
    painerPathRect(&painter);
    setMask(pixmap.mask());
    QWidget::resizeEvent(event);
}

void SelectWidget::handleMouseMoveEvent(QMouseEvent* mouseevent)
{
    mouseMoveEvent(mouseevent);
}

void SelectWidget::handleMousePressEvent(QMouseEvent* mouseevent)
{
    mousePressEvent(mouseevent);
}

void SelectWidget::handleReleaseEvent(QMouseEvent* mouseevent)
{
    mouseReleaseEvent(mouseevent);
}
//控件编辑类
TextEditor::TextEditor(QWidget* parent) 
    : QWidget(parent)
    , m_lineEdit(new QLineEdit(this))
{
    installEventFilter(this);
    m_lineEdit->setObjectName("q_texteditor");
    m_lineEdit->setFrame(false);
    m_lineEdit->setBackgroundRole(parent->backgroundRole());

    setFocusProxy(m_lineEdit);
    connect(m_lineEdit, &QLineEdit::editingFinished, this, &TextEditor::editingFinished);
    connect(m_lineEdit, &QLineEdit::returnPressed, this, &TextEditor::slotEditingFinished);
    connect(m_lineEdit, &QLineEdit::textChanged, this, &TextEditor::slotTextChanged);
}

TextEditor::~TextEditor()
{

}

void TextEditor::slotTextChanged(const QString &text)
{
    m_cachedText = text;
}

void TextEditor::slotEditingFinished()
{
    emit textChanged(m_cachedText);
}

QString TextEditor::text() const{
    return m_cachedText;
}

void TextEditor::setText(const QString &text){
    m_cachedText = text;
    m_lineEdit->setText(text);
}

void TextEditor::setAlignment(Qt::Alignment align)
{
    m_lineEdit->setAlignment(align);
}

void TextEditor::selectAll() {
    m_lineEdit->selectAll();
}

void TextEditor::clear() {
    m_lineEdit->clear();
}

void TextEditor::resizeEvent(QResizeEvent * event) {
    m_lineEdit->resize(event->size());
}

QSize TextEditor::sizeHint() const {
    return  m_lineEdit->sizeHint();
}

QSize TextEditor::minimumSizeHint() const {
    return  m_lineEdit->minimumSizeHint();
}

void TextEditor::installEventFilter(QObject *filterObject)
{
    if (m_lineEdit)
        m_lineEdit->installEventFilter(filterObject);
}

CompontEditor::CompontEditor(QWidget *widget, const QString& text, QWidget* parent)
    : TextEditor(parent)
    , m_widget(widget)
{
    qApp->installEventFilter(this);
    setText(text);
    selectAll();
    setAlignment(alignment());

    QRect r = editRectangle();
    setGeometry(QRect(widget->mapTo(widget->window(), r.topLeft()), r.size()));

    m_selectWidget = new SelectWidget(parent);
    m_selectWidget->setSelectWidget(widget);
    m_selectWidget->installEventFilter(this);
    m_selectWidget->setMinimumSize(m_widget->minimumSize() + QSize(4, 4));
    m_selectWidget->setMaximumSize(m_widget->maximumSize() + QSize(4, 4));
    m_selectWidget->setGeometry(QRect(m_widget->pos() + QPoint(-2, -2), m_widget->size() + QSize(4, 4)));
    this->hide();
    m_selectWidget->hide();

    connect(this, &TextEditor::editingFinished, [this](){
        this->hide();
        m_widget->stackUnder(m_selectWidget);
        m_widget->setProperty("text", this->text());
    });

    connect(m_widget, &QWidget::destroyed, [this](){
        delete m_selectWidget;
    });
}

Qt::Alignment CompontEditor::alignment() const {
    if (m_widget->metaObject()->indexOfProperty("alignment") != -1)
        return Qt::Alignment(m_widget->property("alignment").toInt());

    if (qobject_cast<const QPushButton *>(m_widget) || qobject_cast<const QToolButton *>(m_widget))
        return Qt::AlignHCenter;

    return Qt::AlignJustify;
}

CompontEditor::~CompontEditor(){
    
}

QRect CompontEditor::editRectangle() const
{
    QStyleOptionButton opt;
    opt.init(m_widget);
    if (m_widget->inherits("QPushButton")){
        return m_widget->style()->subElementRect(QStyle::SE_PushButtonContents, &opt, m_widget);
    }
    else if (m_widget->inherits("QRadioButton")){
        return m_widget->style()->subElementRect(QStyle::SE_RadioButtonContents, &opt, m_widget);
    }
    else if (m_widget->inherits("QCheckBox")){
        return m_widget->style()->subElementRect(QStyle::SE_CheckBoxContents, &opt, m_widget);
    }
    else{
        return opt.rect;
    }   
}

bool CompontEditor::eventFilter(QObject *watched, QEvent *event)
{
    if (event->type() == QEvent::MouseButtonDblClick){
        if (watched == m_widget){
            this->selectAll();
            this->setFocus();
            this->show();
            m_selectWidget->stackUnder(m_widget);
        }
    }
    else if (event->type() == QEvent::MouseButtonPress){
        QMouseEvent* mouse = dynamic_cast<QMouseEvent*>(event);
        if (watched != m_widget){
            QRect rect(this->mapToGlobal(QPoint(0, 0)), this->size());
            if (this->isVisible() && !rect.contains(mouse->globalPos())){
                emit editingFinished();
            }
        }
        else{
            m_selectWidget->handleMousePressEvent(dynamic_cast<QMouseEvent*>(event));
        }
    }
    else if (event->type() == QEvent::MouseMove){
        if (watched == m_widget){
            m_selectWidget->handleMouseMoveEvent(dynamic_cast<QMouseEvent*>(event));
        }
    }
    else if (event->type() == QEvent::MouseButtonRelease){
        if (watched == m_widget){
            m_selectWidget->handleReleaseEvent(dynamic_cast<QMouseEvent*>(event));
        }
    }
    else if (event->type() == QEvent::Show) {
        if (watched == this){
            QRect r = editRectangle();
            setGeometry(QRect(m_widget->mapTo(m_widget->window(), r.topLeft()), r.size()));
        }
    }
    return TextEditor::eventFilter(watched, event);
}

QWidget* CompontEditor::getEditorWidget() const
{
    return m_widget;
}

void CompontEditor::clearSelect()
{
    m_selectWidget->hide();
}

void CompontEditor::setSelect()
{
    m_selectWidget->show();
}

bool CompontEditor::isSelect()
{
    return m_selectWidget->isVisible();
}

QWidget* CompontEditor::getSelectWidget() const
{
    return m_selectWidget;
}

工程文件

Qt交流大会 853086607 免费群中


在这里插入图片描述

结尾

不定期上传新作品,解答群中作品相关问题。相关外,能解答则解答。欢迎大家一起探索Qt世界!

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,366评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,521评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 165,689评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,925评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,942评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,727评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,447评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,349评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,820评论 1 317
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,990评论 3 337
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,127评论 1 351
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,812评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,471评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,017评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,142评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,388评论 3 373
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,066评论 2 355

推荐阅读更多精彩内容