void SkeletonTool::togglePinnedStatus(int columnIndex, int frame, bool shiftPressed) { Skeleton skeleton; buildSkeleton(skeleton, columnIndex); if (!skeleton.getRootBone() || !skeleton.getRootBone()->getStageObject()) return; Skeleton::Bone *bone = skeleton.getBoneByColumnIndex(columnIndex); assert(bone); if (!bone) return; TogglePinnedStatusUndo *undo = new TogglePinnedStatusUndo(this, frame); for (int i = 0; i < skeleton.getBoneCount(); i++) { TStageObject *obj = skeleton.getBone(i)->getStageObject(); if (obj) { undo->addBoneId(obj->getId()); obj->setKeyframeWithoutUndo(frame); } } getApplication()->getCurrentXsheet()->notifyXsheetChanged(); getApplication()->getCurrentObject()->notifyObjectIdChanged(false); undo->setOldTemp(m_temporaryPinnedColumns); bool isTemporaryPinned = m_temporaryPinnedColumns.count(columnIndex) > 0; if (shiftPressed || isTemporaryPinned) { if (isTemporaryPinned) m_temporaryPinnedColumns.erase(columnIndex); else m_temporaryPinnedColumns.insert(columnIndex); } else { TXsheet *xsh = TTool::getApplication()->getCurrentXsheet()->getXsheet(); TAffine placement = xsh->getPlacement(bone->getStageObject()->getId(), frame); TStageObjectId rootId = skeleton.getRootBone()->getStageObject()->getId(); TAffine rootPlacement = xsh->getPlacement(rootId, frame); int pinnedStatus = bone->getPinnedStatus(); if (pinnedStatus != Skeleton::Bone::PINNED) { int oldPinned = -1; for (int i = 0; i < skeleton.getBoneCount(); i++) { TStageObject *obj = skeleton.getBone(i)->getStageObject(); if (obj->getPinnedRangeSet()->isPinned(frame)) { oldPinned = i; break; } } int lastFrame = 1000000; if (oldPinned >= 0) { assert(skeleton.getBone(oldPinned) != bone); TStageObject *obj = skeleton.getBone(oldPinned)->getStageObject(); const TPinnedRangeSet::Range *range = obj->getPinnedRangeSet()->getRange(frame); assert(range && range->first <= frame && frame <= range->second); lastFrame = range->second; TPinnedRangeSet *rangeSet = obj->getPinnedRangeSet(); rangeSet->removeRange(frame, range->second); obj->invalidate(); undo->setOldRange(oldPinned, frame, range->second, rangeSet->getPlacement()); } else { for (int i = 0; i < skeleton.getBoneCount(); i++) { TStageObject *obj = skeleton.getBone(i)->getStageObject(); const TPinnedRangeSet::Range *range = obj->getPinnedRangeSet()->getNextRange(frame); if (range) { assert(range->first > frame); if (range->first - 1 < lastFrame) lastFrame = range->first - 1; } } } TStageObject *obj = bone->getStageObject(); TPinnedRangeSet *rangeSet = obj->getPinnedRangeSet(); rangeSet->setRange(frame, lastFrame); if (frame == 0) { // this code should be moved elsewhere, possibly in the stageobject implementation // the idea is to remove the normal TStageObject *rootObj = skeleton.getRootBone()->getStageObject(); rootObj->setStatus(TStageObject::XY); placement = rootObj->getPlacement(0).inv() * placement; rootObj->setStatus(TStageObject::IK); rangeSet->setPlacement(placement); rootObj->invalidate(); } undo->setNewRange(bone->getColumnIndex(), frame, lastFrame, rangeSet->getPlacement()); } } undo->setNewTemp(m_temporaryPinnedColumns); TUndoManager::manager()->add(undo); }