“点一下就能改”——这个功能为首富赚到了多少money?

从 QQ 个性签名编辑看 JavaScript 面向对象编程:一个原生 DOM 的 OOP 实践

本文通过复刻 QQ"点击个性签名即可编辑"的交互效果,手写一个 EditInPlace(就地编辑)组件,深入理解 JavaScript 中的面向对象编程(OOP)思想,并探讨其在现代前端开发中的意义与局限。


🎯 场景还原:QQ 的"就地编辑"体验

打开 QQ 资料页,你会发现个性签名处是这样设计的:

  • 默认显示为一段文本;
  • 点击后,文本变成输入框 + "保存/取消"按钮;
  • 输入内容后点击"保存",内容更新;点击"取消"则还原。

这种 "无需跳转页面、直接在当前位置编辑" 的交互模式,我们称之为 "就地编辑"(Edit in Place) 。它简洁、高效,是 Web 2.0 时代的经典 UI 模式之一。

今天,我们就用 原生 JavaScript + OOP 来实现它!


🛠️ 核心实现:EditInPlace

以下是我们实现的核心代码(已简化注释,保留关键逻辑):

js 复制代码
function EditInPlace(id, value, parentElement) {
  this.id = id;
  this.value = value || '这个家伙很懒,什么都没有留下';
  this.parentElement = parentElement;

  // DOM 元素引用
  this.containerElement = null;
  this.staticElement = null;   // 显示文本的 span
  this.fieldElement = null;    // 可编辑的 input
  this.saveButton = null;
  this.cancelButton = null;

  this.createElement();  // 创建 DOM 结构
  this.attachEvent();    // 绑定事件
}

EditInPlace.prototype = {
  createElement() {
    this.containerElement = document.createElement('div');
    this.containerElement.id = this.id;

    // 文本显示区
    this.staticElement = document.createElement('span');
    this.staticElement.innerHTML = this.value;
    this.containerElement.appendChild(this.staticElement);

    // 编辑输入框
    this.fieldElement = document.createElement('input');
    this.fieldElement.type = 'text';
    this.fieldElement.value = this.value;
    this.containerElement.appendChild(this.fieldElement);

    // 按钮
    this.saveButton = document.createElement('input');
    this.saveButton.type = 'button';
    this.saveButton.value = '保存';
    this.containerElement.appendChild(this.saveButton);

    this.cancelButton = document.createElement('input');
    this.cancelButton.type = 'button';
    this.cancelButton.value = '取消';
    this.containerElement.appendChild(this.cancelButton);

    this.parentElement.appendChild(this.containerElement);
    this.convertToText(); // 初始为只读状态
  },

  convertToText() {
    this.staticElement.style.display = 'inline';
    this.fieldElement.style.display = 'none';
    this.saveButton.style.display = 'none';
    this.cancelButton.style.display = 'none';
  },

  convertToField() {
    this.staticElement.style.display = 'none';
    this.fieldElement.value = this.value;
    this.fieldElement.style.display = 'inline';
    this.saveButton.style.display = 'inline';
    this.cancelButton.style.display = 'inline';
  },

  attachEvent() {
    this.staticElement.addEventListener('click', () => this.convertToField());
    this.saveButton.addEventListener('click', () => this.save());
    this.cancelButton.addEventListener('click', () => this.cancel());
  },

  save() {
    const value = this.fieldElement.value.trim();
    this.value = value === '' ? '这个家伙很懒,什么都没有留下' : value;
    this.staticElement.innerHTML = this.value;
    this.convertToText();
    // TODO: 可在此处调用 API 保存到服务器
  },

  cancel() {
    this.convertToText();
  }
};

✅ 使用方式

html 复制代码
<div id="app"></div>

<script>
  new EditInPlace('signature', 'Hello World!', document.getElementById('app'));
</script>

🧠 OOP 设计亮点解析

这段代码虽小,却完整体现了 面向对象编程的四大核心思想

1. 封装(Encapsulation)

  • 所有状态(value、DOM 引用)和行为(saveconvertToText)都封装在 EditInPlace 实例中;
  • 外部只需调用构造函数,无需关心内部 DOM 如何创建或切换。

2. 抽象(Abstraction)

  • 用户看到的是"可编辑的签名",而隐藏了"span/input 切换"、"事件绑定"等细节;
  • 提供清晰的接口:点击 → 编辑 → 保存/取消。

3. 单一职责

  • createElement 只负责构建 DOM;
  • attachEvent 只负责绑定事件;
  • save 只负责处理保存逻辑。
  • 各司其职,便于维护和测试。

4. 可复用性

  • 只要传入不同的 idvalue 和挂载点,就能创建多个独立的编辑器实例;
  • 无全局污染,实例之间互不影响。

✅ 总结

"就地编辑"看似简单,却是 OOP 与 DOM 操作的绝佳练兵场。

通过 EditInPlace,我们不仅复刻了 QQ 的经典交互,更深入理解了:

  • 如何用对象封装状态与行为;
  • 如何组织可维护的原生 JS 代码;
相关推荐
Jingyou38 分钟前
JavaScript 实现深拷贝
前端·javascript
编程猪猪侠38 分钟前
Vue 通用复选框组互斥 Hooks:兼容 Element Plus + Ant Design Vue
前端·javascript·vue.js
凡人程序员38 分钟前
搭建 monorepo 项目
前端·javascript
linda261838 分钟前
说说 Map 和 Set 的区别及实际应用
前端·javascript
小飞侠在吗41 分钟前
vue setup与OptionsAPI
前端·javascript·vue.js
疯不皮41 分钟前
tiptiap3如何实现编辑器内部嵌套多个富文本编辑器
前端·vue.js·开源
溪饱鱼42 分钟前
主动与被动AI交互范式
前端·后端·aigc
我叫黑大帅43 分钟前
如何实现UniApp登录拦截?
前端·javascript·vue.js
写代码的皮筏艇44 分钟前
Sequelize 详细指南
前端·后端