المنزل الذكي المنزل الذكي

تستمر وحدات Master Kit في مشاركة الحلول معنا :
"كتجربة ، قررت هنا أن أحاول صنع نوع من" المنزل الذكي "النموذجي على عجل وبأقل تكلفة. كان هناك الكثير من أصحاب الفنادق: الضوء والتهوية والنوافذ والمياه والتحكم بالأشعة تحت الحمراء للأجهزة الكهربائية. في البداية ، قررت أن أقتصر على الحد الأدنى من المهام: التهوية والإضاءة في الغرفة.



اعتمد النظام على Arduino Uno ، مع القدرة على التحكم في أربعة دبابيس بشكل مستقل عن بعضها البعض ، والعديد من وحدات التحكم اللاسلكية من Master Kit: اتخذت مرحلات MP3328 و MP3330 ذات القناة الواحدة دور الأجهزة التنفيذية ، ويتم إرسال الإشارات إليها باستخدام جهاز إرسال من ثماني قنوات MP3329 بتردد 433 ميجاهرتز. في MP3330 ، قمت بتعليق التحكم في شريطي LED على الأريكة ، - إضاءة خلفية مريحة للقراءة المسائية ، وعلى MP3328 - التحكم في ماكينة مؤازرة لفتح / إغلاق النافذة. قمت ببناء هيكل محرك الأقراص من المواد المرتجلة ، أي من أجزاء مصمم LEGO.











حسنًا ، وبالطبع ، لم يكن التحكم في الراديو البسيط كافيًا بالنسبة لي: كنت بحاجة إلى الوصول إلى النظام بأكمله من خلال واجهة الويب ومراقبة الحالة في الوقت الفعلي.

كمورد تخزين ، أخذت MySQL قياسيًا ، حيث أنشأت جدولًا يحتوي على معرفات وحدات التحكم والدبابيس وقيمة الحالة لكل منها ، لأغراض أمنية ، وربط معرف نظام مشترك لكل منها. أيضًا ، لقد تمكنت بشدة من تعيين القدرة على ضبط مؤقت ثابت لتغيير الحالة ، ويتم أيضًا توفير حقل لهذا.



عند العمل مع مآخذ الويب ، كان Arduino عربات التي تجرها الدواب قليلاً ، ولم يكن هناك وقت لمعرفة الأسباب ، لذلك قمت برسمها بطريقة بسيطة: مع طلبات ajax كل 100 مللي ثانية. هذا بالطبع كارثي لحركة المرور (حوالي 30 ميغابايت في اليوم) ، ولكن بالنسبة لشقة ذات إنترنت غير محدود للمرة الأولى ، هذا يكفي.

أبسط واجهة ويب: 4 أزرار ، واحد لكل دبوس. تؤدي ضغطة واحدة إلى تغيير الحالة ، وتضغط ضغطة طويلة على المؤقت (لدي 3 دقائق حتى الآن) لتغييره.



ونتيجة لذلك ، فإن النسخة المبسطة للمنطق هي كما يلي:

تصل واجهة الويب إلى الخادم وتتحقق من حالة وحدات التحكم ، وبعد ذلك تعرض الواجهة للعمل مع النظام: 4 أزرار في الحالة المقابلة:

نصوص
$.ajax({
			url: 'engine/ajax.php',
			type: 'POST',
			dataType: 'json',
			data: {action: 'getStates'},
		})
		.done(function(data) {
			for (key in data.success){
				var controller = data.success[key];
				if($('.btn#btn_' + controller.id).length === 0){
					html = '';
					html += '<div ';
					if (controller.timer_switch > 0){
						var seconds = 60  - (controller.timer_switch % 60)
						html += 'seconds="' + seconds + '"';
					}
					html += 'id="btn_' + controller.id + '" class="btn" state="' + controller.state + '"><div class="btn_timer"></div></div>';
					$('#btn_placer').append(html);
				} else {
					if (controller.state == 1){
						$('.btn#btn_' + controller.id).removeClass('btn_off').addClass('btn_on');
					} else if (controller.state == 0){
						$('.btn#btn_' + controller.id).removeClass('btn_on').addClass('btn_off');
					}
					if (controller.timer_switch > 0){
						var seconds = 60 - controller.timer_switch % 60;
						var minutes = Math.floor(controller.timer_switch / 60);
						$('.btn#btn_' + controller.id).addClass('seconds').css('background-position', (-seconds * 100) + 'px 0px').find('.btn_timer').text(minutes + 'M');
					} else {
						$('.btn#btn_' + controller.id).css('background-position', '0px 0px').removeClass('seconds').find('.btn_timer').text('');
					}
				}
			}
			setTimeout(function(){
				getStates();
			}, 1000);
		})
		.fail(function(data) {
			console.log('error');
		});



function getStates($sql){
	$result = $sql->query("SELECT * FROM `controllers` WHERE `home_id` = '1' ORDER BY `order`");
	if (isset($result->rows)){
		$result = $result->rows;
		foreach ($result as $key => $value) {
			if (strtotime($result[$key]['timer']) > -62169990000){
				// echo strtotime($result[$key]['timer']);
				$timer_switch = strtotime($result[$key]['timer']) - strtotime(date("Y-m-d H:i:s"));
				$result[$key]['timer_switch'] = $timer_switch;
				if ($timer_switch < 0){
					$sql->query("UPDATE `controllers` SET `state` = '".$result[$key]['timer_state']."', `timer` = '0000-00-00 00:00:00', `timer_state` = '' WHERE `id` = '".$result[$key]['id']."'");
				}
			}
		}
		$res['success'] = $result;
	} else {
		$res['error'] = ' ';
	}
	return $res;
}




عند الضغط على الزر المقابل ، يتم إرسال طلب POST إلى ملف ajax.php:

جافا سكريبت
$(document).on('mousedown', '.btn', function(event){
			event.preventDefault();
			var id = parseInt($(this).attr('id').replace('btn_', ''));
			click_wait = false;
			mousetimer = setTimeout(function(){
				click_wait = true;
				setTimer(id);	
			}, 2000);
		});

		$(document).on('mouseup', '.btn', function(){
			clearTimeout(mousetimer);
			if (!click_wait){
				var id = parseInt($(this).attr('id').replace('btn_', ''));
				switchController(id);
				console.log('click !!!');
				click_wait = false;
			}
		});
	function switchController(id){
		var el = $('.btn#btn_' + id);
		var state = parseInt($(el).attr('state'));
		var need_state;
		if (state == 0){
			need_state = 1;
		} else if (state == 1){
			need_state = 0;
		}
		$(el).addClass('waiting');
		$.ajax({
			url: 'engine/ajax.php',
			type: 'POST',
			dataType: 'json',
			data: {action: 'setState', id: id, state: state, need_state: need_state},
		})
		.done(function(data) {
			if (data.success == 'ok'){
				$(el).attr('state', need_state);
				$(el).removeClass('waiting').removeClass('btn_on').removeClass('btn_off');
				if(need_state == 1){
					$(el).addClass('btn_on');
				} else if(need_state == 0){
					$(el).addClass('btn_off');
				}
			}
		})
		.fail(function(data) {
			console.log('error');
		});
	}
	function setTimer(id){
		var el = $('.btn#btn_' + id);
		var state = parseInt($(el).attr('state'));
		var need_state;
		if (state == 0){
			need_state = 1;
		} else if (state == 1){
			need_state = 0;
		}
		$(el).attr('seconds', 0);
		$(el).addClass('seconds');
		$.ajax({
			url: 'engine/ajax.php',
			type: 'POST',
			dataType: 'json',
			data: {action: 'setTimer', id: id, state: state, need_state: need_state},
		})
		.done(function(data) {
		})
		.fail(function(data) {
			console.log('error');
		});
	}




في معرف المتغير ، نقوم بتمرير معرف وحدة التحكم ، وفي حالة المتغير ، قيمة حالته. في ملف ajax.php ، نحصل على طلب POST ونضع البيانات الجديدة في قيم التسجيل للمعرف المناظر.

بي أتش بي
function setState($sql){
	$need_state = (int)$_POST['need_state'];
	$state = (int)$_POST['state'];
	$id = (int)$_POST['id'];

	$result = $sql->query("UPDATE `controllers` SET `state` = '".$need_state."', `timer` = '0000-00-00 00:00:00' WHERE `id` = '".$id."'");
	//   Arduino
	$result = $sql->query("SELECT `state` FROM `controllers` WHERE `id` = '".$id."'");
	if ($need_state == $result->row['state']){
		$res['success'] = 'ok';
	} else {
		$res['success'] = 'err';
	}
	return $res;
}

function setTimer($sql){
	$need_state = (int)$_POST['need_state'];
	$state = (int)$_POST['state'];
	$id = (int)$_POST['id'];

	$result = $sql->query("UPDATE `controllers` SET `timer` = NOW() + INTERVAL 3 MINUTE, `timer_state` = '".$need_state."' WHERE `id` = '".$id."'");
	//   Arduino
	$result = $sql->query("SELECT `state` FROM `controllers` WHERE `id` = '".$id."'");
	if ($need_state == $result->row['state']){
		$res['success'] = 'ok';
	} else {
		$res['success'] = 'err';
	}
	return $res;
}




Arduino ، بدوره ، يصل إلى خادم الويب عبر إيثرنت ، ويتحقق من مفتاح النظام الفريد ، وقيم وحدة التحكم ، ويوزع السلسلة المستلمة ويبدل قيم الدبوس.

بي أتش بي
if (isset($_GET['key']) && $_GET['key'] !== ''){
		$key = $_GET['key'];
	} else {
		die;
	}
	
$key = $sql->escape($key);
	$result = $sql->query("SELECT * FROM `controllers` WHERE `home_id` = '1' AND `key` = '".$key."' ORDER BY `order`");
	foreach ($result->rows as $key => $value) {
		if ($value['id'] == '1'){
			if ($value['state'] == '1'){
				$ar .= 'Q';
			} else if ($value['state'] == '0') {
				$ar .= 'q';
			}
		}
		if ($value['id'] == '2'){
			if ($value['state'] == '1'){
				$ar .= 'W';
			} else if ($value['state'] == '0') {
				$ar .= 'w';
			}
		}
		if ($value['id'] == '3'){
			if ($value['state'] == '1'){
				$ar .= 'E';
			} else if ($value['state'] == '0') {
				$ar .= 'e';
			}
		}
	}




بعد ذلك ، يتم نقل سطر من النموذج "٪ qUerTY" إلى رسم تخطيطي حيث يتم تحليله ، اعتمادًا على القواعد الشائكة: كل حرف يتوافق مع رقم التعريف الشخصي الخاص به ، والسجل مسؤول عن القيمة النهائية: رأس المال - 1 ، صغيرة - 0.

اردوينو
void readData(){
  if (led_connect){
    digitalWrite(6, HIGH);
  } else {
    digitalWrite(6, LOW);
  }
  digitalWrite(4, LOW);
  previousMillis = currentMillis;
  
  led_connect = !led_connect;
    while (client.available()){      
      switch (char c = client.read()) {
        case 'Q':
          digitalWrite(9, HIGH);
          break;
        case 'q':
          digitalWrite(9, LOW);
          break;
        case 'W':
          digitalWrite(8, HIGH);
          break;
        case 'w':
          digitalWrite(8, LOW);
          break;
        case 'E':
          digitalWrite(7, HIGH);
          myservo.write(0);
          break;
        case 'e':
          digitalWrite(7, LOW);
          myservo.write(180);
          break;
      }
    }
}



يمكنك نقل القيم من 0 إلى 180 (درجات الاستدارة) إلى الدبوس المسؤول عن آلة المؤازرة ، ولنقل إلى المعتم - حتى 255. لا يزال لدي قيمتان: 0 أو 1.





لم تكن هناك مشاكل في توصيل الوحدات: كلاهما يتم إقران المرحلات بجهاز الإرسال مرة واحدة ، دون الحاجة إلى إعادة التجليد ، وأداء مهمتها بشكل مثالي.





ولكن مع قوة آلة المؤازرة ، أخطأت في الحساب قليلاً: لقد أخذت 6 كجم ، وهو ما يكفي بشكل عام لتغطية وترك نافذة الوشاح ، التي فتحت تحت وزنها ، ولكن من الأفضل أن تأخذ شيئًا أكثر قوة ، على حالة الرياح القوية أو الغاطسة.

بشكل عام ، مجال الإبداع غير محدود: على سبيل المثال ، يمكنك توصيل أجهزة الاستشعار ، على سبيل المثال ، ثاني أكسيد الكربون أو درجة الحرارة ، وفتح / إغلاق النافذة عند الوصول إلى مؤشرات معينة.
حسنًا ، بالطبع ، أخطط لإعادة كتابة النظام بأكمله على مآخذ الويب ، مما يجعل الطلبات أكثر سهولة - وجعل نقل قيم المعلمات وتحليلها أكثر مرونة.

رسم لخلفية اردوينو

فيديو

العمل الخلفي


فتح النافذة


دميتري كوزنتسوف "

Source: https://habr.com/ru/post/ar383679/


All Articles