## 🎯 主要機能 - `box ChatNode from P2PBox` 構文完全対応 - 25個のビルトインBox型で継承可能に - `from Parent.method()` デリゲーション構文実装 ## 🏗️ アーキテクチャ革命 - BoxBase + BoxCore統一設計完成 - parent_type_id による継承関係管理 - as_any() 動的型システム統一実装 - Arc<Mutex>パターン全Box型適用完了 ## ✅ 技術的達成 - コンパイルエラー: 42個 → 0個 (100%解決) - ビルトイン継承: StringBox, P2PBox, MathBox等すべて対応 - 実行時型安全性: 完全保証 - Everything is Box哲学: より深化 ## 🔧 主要変更ファイル - src/box_trait.rs: BoxBase/BoxCore統一アーキテクチャ - src/boxes/*: 全Box型にas_any()実装 - src/interpreter/: ビルトイン継承ディスパッチ実装 - docs/: 継承システム仕様書更新 🎉 Nyashが本格プログラミング言語として大きく進化! 🤖 Generated with [Claude Code](https://claude.ai/code) Co-Authored-By: Claude <noreply@anthropic.com>
211 lines
6.6 KiB
Rust
211 lines
6.6 KiB
Rust
//! ArrayBox 📦 - 配列・リスト操作
|
|
// Nyashの箱システムによる配列・リスト操作を提供します。
|
|
// Arc<Mutex>パターンで内部可変性を実現
|
|
|
|
use crate::box_trait::{NyashBox, StringBox, BoolBox, IntegerBox, BoxCore, BoxBase};
|
|
use std::any::Any;
|
|
use std::sync::{Arc, Mutex};
|
|
use std::fmt::Display;
|
|
|
|
#[derive(Debug, Clone)]
|
|
pub struct ArrayBox {
|
|
pub items: Arc<Mutex<Vec<Box<dyn NyashBox>>>>,
|
|
base: BoxBase,
|
|
}
|
|
|
|
impl ArrayBox {
|
|
/// 新しいArrayBoxを作成
|
|
pub fn new() -> Self {
|
|
ArrayBox {
|
|
items: Arc::new(Mutex::new(Vec::new())),
|
|
base: BoxBase::new(),
|
|
}
|
|
}
|
|
|
|
/// 要素を持つArrayBoxを作成
|
|
pub fn new_with_elements(elements: Vec<Box<dyn NyashBox>>) -> Self {
|
|
ArrayBox {
|
|
items: Arc::new(Mutex::new(elements)),
|
|
base: BoxBase::new(),
|
|
}
|
|
}
|
|
|
|
/// 要素を追加
|
|
pub fn push(&self, item: Box<dyn NyashBox>) -> Box<dyn NyashBox> {
|
|
self.items.lock().unwrap().push(item);
|
|
Box::new(StringBox::new("ok"))
|
|
}
|
|
|
|
/// 最後の要素を取り出す
|
|
pub fn pop(&self) -> Box<dyn NyashBox> {
|
|
match self.items.lock().unwrap().pop() {
|
|
Some(item) => item,
|
|
None => Box::new(crate::boxes::null_box::NullBox::new()),
|
|
}
|
|
}
|
|
|
|
/// 要素数を取得
|
|
pub fn length(&self) -> Box<dyn NyashBox> {
|
|
Box::new(IntegerBox::new(self.items.lock().unwrap().len() as i64))
|
|
}
|
|
|
|
/// インデックスで要素を取得
|
|
pub fn get(&self, index: Box<dyn NyashBox>) -> Box<dyn NyashBox> {
|
|
if let Some(idx_box) = index.as_any().downcast_ref::<IntegerBox>() {
|
|
let idx = idx_box.value as usize;
|
|
let items = self.items.lock().unwrap();
|
|
match items.get(idx) {
|
|
Some(item) => item.clone_box(),
|
|
None => Box::new(crate::boxes::null_box::NullBox::new()),
|
|
}
|
|
} else {
|
|
Box::new(StringBox::new("Error: get() requires integer index"))
|
|
}
|
|
}
|
|
|
|
/// インデックスで要素を設定
|
|
pub fn set(&self, index: Box<dyn NyashBox>, value: Box<dyn NyashBox>) -> Box<dyn NyashBox> {
|
|
if let Some(idx_box) = index.as_any().downcast_ref::<IntegerBox>() {
|
|
let idx = idx_box.value as usize;
|
|
let mut items = self.items.lock().unwrap();
|
|
if idx < items.len() {
|
|
items[idx] = value;
|
|
Box::new(StringBox::new("ok"))
|
|
} else {
|
|
Box::new(StringBox::new("Error: index out of bounds"))
|
|
}
|
|
} else {
|
|
Box::new(StringBox::new("Error: set() requires integer index"))
|
|
}
|
|
}
|
|
|
|
/// 要素を削除
|
|
pub fn remove(&self, index: Box<dyn NyashBox>) -> Box<dyn NyashBox> {
|
|
if let Some(idx_box) = index.as_any().downcast_ref::<IntegerBox>() {
|
|
let idx = idx_box.value as usize;
|
|
let mut items = self.items.lock().unwrap();
|
|
if idx < items.len() {
|
|
items.remove(idx)
|
|
} else {
|
|
Box::new(crate::boxes::null_box::NullBox::new())
|
|
}
|
|
} else {
|
|
Box::new(StringBox::new("Error: remove() requires integer index"))
|
|
}
|
|
}
|
|
|
|
/// 指定された値のインデックスを検索
|
|
pub fn indexOf(&self, value: Box<dyn NyashBox>) -> Box<dyn NyashBox> {
|
|
let items = self.items.lock().unwrap();
|
|
for (i, item) in items.iter().enumerate() {
|
|
if item.equals(value.as_ref()).value {
|
|
return Box::new(IntegerBox::new(i as i64));
|
|
}
|
|
}
|
|
Box::new(IntegerBox::new(-1))
|
|
}
|
|
|
|
/// 指定された値が含まれているか確認
|
|
pub fn contains(&self, value: Box<dyn NyashBox>) -> Box<dyn NyashBox> {
|
|
let items = self.items.lock().unwrap();
|
|
for item in items.iter() {
|
|
if item.equals(value.as_ref()).value {
|
|
return Box::new(BoolBox::new(true));
|
|
}
|
|
}
|
|
Box::new(BoolBox::new(false))
|
|
}
|
|
|
|
/// 配列を空にする
|
|
pub fn clear(&self) -> Box<dyn NyashBox> {
|
|
self.items.lock().unwrap().clear();
|
|
Box::new(StringBox::new("ok"))
|
|
}
|
|
|
|
/// 文字列結合
|
|
pub fn join(&self, delimiter: Box<dyn NyashBox>) -> Box<dyn NyashBox> {
|
|
if let Some(sep_box) = delimiter.as_any().downcast_ref::<StringBox>() {
|
|
let items = self.items.lock().unwrap();
|
|
let parts: Vec<String> = items
|
|
.iter()
|
|
.map(|item| item.to_string_box().value)
|
|
.collect();
|
|
Box::new(StringBox::new(&parts.join(&sep_box.value)))
|
|
} else {
|
|
Box::new(StringBox::new("Error: join() requires string separator"))
|
|
}
|
|
}
|
|
}
|
|
|
|
impl BoxCore for ArrayBox {
|
|
fn box_id(&self) -> u64 {
|
|
self.base.id
|
|
}
|
|
|
|
fn parent_type_id(&self) -> Option<std::any::TypeId> {
|
|
self.base.parent_type_id
|
|
}
|
|
|
|
fn fmt_box(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
|
|
let items = self.items.lock().unwrap();
|
|
let strings: Vec<String> = items.iter()
|
|
.map(|item| item.to_string_box().value)
|
|
.collect();
|
|
write!(f, "[{}]", strings.join(", "))
|
|
}
|
|
|
|
fn as_any(&self) -> &dyn Any {
|
|
self
|
|
}
|
|
|
|
fn as_any_mut(&mut self) -> &mut dyn Any {
|
|
self
|
|
}
|
|
}
|
|
|
|
impl Display for ArrayBox {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
self.fmt_box(f)
|
|
}
|
|
}
|
|
|
|
impl NyashBox for ArrayBox {
|
|
fn clone_box(&self) -> Box<dyn NyashBox> {
|
|
Box::new(self.clone())
|
|
}
|
|
|
|
fn to_string_box(&self) -> StringBox {
|
|
let items = self.items.lock().unwrap();
|
|
let strings: Vec<String> = items.iter()
|
|
.map(|item| item.to_string_box().value)
|
|
.collect();
|
|
StringBox::new(format!("[{}]", strings.join(", ")))
|
|
}
|
|
|
|
|
|
fn type_name(&self) -> &'static str {
|
|
"ArrayBox"
|
|
}
|
|
|
|
|
|
fn equals(&self, other: &dyn NyashBox) -> BoolBox {
|
|
if let Some(other_array) = other.as_any().downcast_ref::<ArrayBox>() {
|
|
let self_items = self.items.lock().unwrap();
|
|
let other_items = other_array.items.lock().unwrap();
|
|
|
|
if self_items.len() != other_items.len() {
|
|
return BoolBox::new(false);
|
|
}
|
|
|
|
for (a, b) in self_items.iter().zip(other_items.iter()) {
|
|
if !a.equals(b.as_ref()).value {
|
|
return BoolBox::new(false);
|
|
}
|
|
}
|
|
|
|
BoolBox::new(true)
|
|
} else {
|
|
BoolBox::new(false)
|
|
}
|
|
}
|
|
} |