Author Archive

X++ :24- Extension ile Standart Bir Tabloya Metot eklemek.

Bu yazıda Visual Studio üzerinden Extension mantığıyla standart bir tabloya yeni bir metot nasıl eklenir anlatmaya çalışacağım.

Önceki yazımda smmActivities’e bir alan eklemiştik. Şimdi yine smmActivities’e bir FDActivityTable kaydı oluşturan bir metot ekleyelim. Öncelikle bir sınıf oluşturalım. Sonrasında aşağıdaki kodu yazalım.

[ExtensionOf(Tablestr(smmActivities))] // Burası önemli bu sınıfın smmActivities tablosunun bir Extension’ı olduğunu söylüyor.

final class FDSmmActivities_Extension // Final olması gerekiyor ve _Extension ile bitirmek lazım.

{

public void createFDActivityTable(smmActivities _smmActivities) // Kayıt oluşturan metodumuz.

{

FDActivityTable       activityTable;

smmActivities           smmActivitiesUp;

NumberSeq               numberSeq;

FDActivityId              activityId;

;

ttsbegin;

numberSeq   = numberSeq::newGetNum(NumberseqmoduleFDActivity::numRefActivityId());

activityId       = numberSeq.num();

activityTable.clear();

activityTable.ActivityId         = activityId;

activityTable.ActivityDate   = DateTimeUtil::date( _smmActivities.ActualEndDateTime);

activityTable.Employee        = _smmActivities.ResponsibleWorker;

activityTable.insert();

smmActivitiesUp                                    = smmActivities::findWithRecId(_smmActivities.RecId , true);

smmActivitiesUp.FDActivityId           = activityTable.ActivityId;

smmActivitiesUp.update();

ttscommit;

}

}

Artık bu metodu smmActivities’te kullanabiliriz. Bir job yazıp deneyebiliriz.

class FDJob3

{

public static void main(Args _args)

{

smmActivities smmActivities;

;

Select firstonly smmActivities;

smmActivities.createFDActivityTable(smmActivities);

}

}

Bu yazıda Extension mantığıyla standart bir tabloya nasıl metot ekleyebileceğinizi anlattım. Burada dikkat etmeniz gereken bir konu bu sınıfların isimlendirmesi ve tekilleştirme.

Selamlar.

www.fatihdemirci.net

TAGs: Microsoft Life Cycle Services, LCS, Azure, Azure DevOps, Extension, Microsoft Dynamics 365, MsDyn365FO, MsDyn365CE, MsDyn365

X++ :23- Extension ile Standart Bir Forma Alan eklemek.

Bu yazıda Visual Studio üzerinden Extension mantığıyla standart bir tabloya eklenen alanların nasıl forma eklenebileceğini anlatmaya çalışacağım.

Bir önceki yazımda smmActivities’e bir alan eklemiştik. Şimdi AOT üzerinden Formları açıyoruz. Sonrasında smmActivities formuna sağ tıklayıp Create extension’a tıklıyoruz.

Resim-1

Projemde Extension oluştu. Adını değiştirebilirim ama benim için şu anada adı uygun. Çift tıklayıp açıyorum.

Resim-2

Açılan Extension formu içinden veri kaynağına baktığımda bir önceki yazımda eklediğim alan otomatik geldi.  Sürükleyip tasarımda uygun bir yere bırakıyorum.

Resim-3

Artık hazırız. Projeyi Build edip çalıştırabiliriz.

Resim-4

Bu yazıda Extension ile standart bir tabloya eklenen yeni bir alanın formda nasıl gösterilebileceğini anlattım. Sonraki yazılarımda Extension ile kod yazmak ile ilgili örnekler yapacağım.

Selamlar.

www.fatihdemirci.net

TAGs: Extension, X++, Microsoft Dynamics 365, MsDyn365FO, MsDyn365CE, MsDyn365

X++ :22- Extension ile Standart Bir Tabloya Alan eklemek.

Bu yazıda Visual Studio üzerinden Extension mantığıyla standart tablolara alan nasıl eklenir anlatmaya çalışacağım.

Öncelikle bu yazılarda Youtube videolarımda oluşturduğum Activity Managment projesini kullanacağım. Bu projede kullanmak üzere aktivitelerimi smmActivities tablosuna bağlamak için bir alan açıp bunu kullanacağız. smmActivities CaseManagment paketinden eğer yoksa öncelikle kendi paketinize eklemiz gerekiyor. Sonrasında sağ tıkladığınızda Create extension’a tıklıyoruz.

Resim-1

Projemde Extension oluştu. Adını değiştirebilirim ama benim için şu anada adı uygun. Çift tıklayıp açıyorum.

Resim-2

Açılan Extension içinden Fields2a sağ tıklayıp yeni bir alan oluşturuyorum.

Resim-3

Alanın özelliklerini belirleyip kaydediyorum.

Resim-4

Projemi eşitliyorum. Eğer hata yoksa tabloya alanı eklemiş oldu.

Resim-5

Bu yazıda Extension ile standart bir tabloya yeni bir alan nasıl eklenir anlatmaya çalıştım. Burada Extension isimlendirmesine dikkat etmek lazım. Birden çok geliştirme yapan varsa özellikle isimlendirme çok önemli oluyor. Bu şekilde alanı ekledikten sonra kodda tablonun standart bir alanı gibi kullanabilirsiniz. Sonraki yazılarımda Extension ile forma alanı eklemek ve kod düzenlemek ile ilgili örnekler yapacağım.

Selamlar.

www.fatihdemirci.net

TAGs: Extension, X++, Microsoft Dynamics 365, MsDyn365FO, MsDyn365CE, MsDyn365

Ax2012’den Dynamics 365 Kod Yükseltmesi Nasıl Yapılır? –3

Bu yazıda Ax2012’den Dynamics 365 Finance and Operations’a nasıl kod güncellemesi yapıyoruz anlatmaya devam edeceğim. İkinci yazıda ModelStore’u yüklemiş ve raporların ve Azure DevOps projesinin oluşturulmasını sağlamıştık. Bu yazıda oluşan kodlara inceleyeceğiz ve VisualStudio’da neler yapmak gerekiyor anlatmaya çalışacağım.

Oluşturduğumuz Azure DevOps projesinde 2 klasör oluşuyor. Aslında ikisinin altında yanı kodlar var. Sırayla inceleyelim.

Resim-1

Trunk altında AplicationSuite’yi açınca sizin Ax2012 olan modellerinizin her biri için klasörler göreceksiniz. Her bir klasör altında Delte klasörü var ve bu klasördeki nesnelerde çakışmalar olduğu için elle düzeltilmesi gerekiyor. Asıl güncelleme eforunun harcanacağı yerler buralar olacak. Bunu dışında kalan nesnelerde sorun yok demektir. Bunları kendi oluşturduğumuz pakete dosya sisteminden kopyalamalısınız.

Resim-2

Visual Studio Source Control Explorer açıp Azure devops projenize bağlanın. Burada sıfır bir development makinesi deploy etmek daha faydalı olacaktır ama şart değil. Ben elimde olan bir makineden bağlandım. Tüm kodları buradan da görebilirsiniz.

Resim-3

Örneğin burada Delta altın çakışma olan nesnelerden bazıları var. Tek tek elden geçmesi lazım. Eğer bir metot eklendiyse bunların Extension mantığına çekilip Delta altından bu nesnenin silinmesi gerekiyor. Eğer kod arasına girilmişse durum biraz daha karışık inceleyip neler yapılması gerektiğine karar verilmesi gerekecek. Bunların ayrıntısına ileriki yazılarımda değineceğim.

Resim-4

Kodları indirmek için bir klasör oluşturup bağlantısını kuruyorum. Bu sayede kendi makineme kodları indirmiş olacağım ve Xml üzerinden kopyalama ve upgrade işlemlerine başlayabilirim.

Resim-5

Bu şamdan sonra Paket yapıma karar verip ona göre paketlerimi oluşturmalıyım. Ax2012 de bulunan model yapımı korumak istersem her model için bir paket oluşturmak bir seçenek olabilir. Burada aklınızda bulunsun çok fazla pakete bölerseniz ve birbirlerine bağımlılıkları varsa bunların sürekli referans hataları alacaksınız. Temel mantık şu olmalı ben bu paketi tek başına alıp bir ortama veya başka bir yere yüklemem gerekir değiniz de bölmek doğru bir yaklaşım olur. Paketleri karar verip modelinizi oluşturduktan sonra Main altındaki upgrade edilmiş bütün Xml leri kendi paketinize kopyalayın. Deltaları kopyalamayın onları tek tek nasıl çözeceğinize ayrıca değineceğim.

Böyle çakışmaları çözmek için hazır hale gelmiş olduk. Eğer birden çok yazılımcı çalışacaksa iş bölümü yapmak lazım. Örneğin Nesneler paylaşılabilir. Enum, EDT, Class ve Table gibi veya modelleri ayırdıysanız model model paylaşabilirsiniz. Burada aynı nesne için iki yazılımcının efor harcamaması için koordine olmak şart.

Bu yazıda çakışmaları çözmeden önceki son aşamaya gelene kadar neler yapılmalı anlatmaya çalıştım. Tabi burada anlattıklarım benim tecrübelerim farklı ekipler farklı yaklaşımları benimseyebilir. İleriki yazarlarımda nene tiplerine göre çakışmalar nasıl çözülür anlatmaya çalışacağım.

Selamlar.

www.fatihdemirci.net

TAGs: Microsoft Life Cycle Services, LCS, Azure, Azure DevOps, Ax2012 Code Upgrade, Microsoft Dynamics 365, MsDyn365FO, MsDyn365CE, MsDyn365

Ax2012’den Dynamics 365 Kod Yükseltmesi Nasıl Yapılır? – 2

Önceki yazımda Ax2012’yi Dynamics 365’e yükseltmek için gerekli ilk adımları anlatmıştım. Bu yazıda dışarı aktardığımız ModelStore dosyasını LCS yükleyip işlemeyi anlatacağım.

Oluşturduğumuz projeyi açıp code upgrade kutucuğuna tıklıyoruz.

Resim-1

Açılan sayfadan Add ile işlemi başlatıyoruz.

Resim-2

Upgrade aşamalarının olduğu bir sayfa açılacak. İşleme tekil bir isim vermek gerekiyor. Hangi versiyondan hangi versiyona taşıma yapılacaksa onu seçiyoruz. Eğer sadece raporu görmek istiyorsanız Estimation Only’i işaretleyebilirsiniz. Bu durumda sadece raporları oluşturacak Azure Devops ta bir işlem yapmayacak.

Resim-3

Add file ile Model Store dosyamızı yüklüyoruz. Burada dosyanın Zip olması gerekiyor Rar kabul etmiyor. Sonrasında Analyze code deyip işlemi başlatıyoruz.

Resim-4

İşlemler tamamlandıktan sonra böyle listelenecek. Name kısmına tıklayıp içine girebilirsiniz.

Resim-5

Tamamlanan işlem sonunda böyle bir sayfa çıkacak. Burada bazı raporlar ve özet bilgiler var.

Resim-6

Burada sizin eforunuzu hesaplamak için kullanacağınız rapor TaskList olacak. Buradaki maddelerin tümüne dokunmanız gerekecek.

Resim-7

Özet olarak yapılan ve kalan işleri bu rapordan görebilirsiniz.

Resim-8

Bu yazıda güncelleme işlemin kod yükseltme adımını anlatmaya çalıştım. Bu aşamadan önce kodları Ax2012 gözden geçirmek ve temiz bir halde ModelStore oluşturup yüklemek bu aşamadan sonra harcanması gereken eforları azaltacaktır. Kod yükseltmeyi gözünüzde büyütmeyin birçok kolaylık var. En azından bu raporları oluşturacak adımları yapıp nasıl bir efor gerektiğini öğrenebilirsiniz. Sonraki yazımda Azure Devops‘ta oluşan kodları ve orada yapılması gereken işleri anlatacağım.

Selamlar.

www.fatihdemirci.net

TAGs: Microsoft Life Cycle Services, LCS, Azure, Azure DevOps, Ax2012 Code Upgrade, Microsoft Dynamics 365, MsDyn365FO, MsDyn365CE, MsDyn365

Dynamics 365 Finance and Operations Self-Service Ortamlarda Canlıya Geliştirme Nasıl Taşınır?

Bu yazıda Dynamics 365 Finance and Operations Environment Type Self-service olan canlı ortamlara geliştirme nasıl taşınır anlatacağım. Şu yazımda Microsot-managed bir ortamda nasıl taşınacağını anlatmıştım. Birkaç adımda farklılık var bunları göstereceğim. Ortamın tipi nasıl belirleniyor derseniz bu lisan sayınız be büyüklüğünüzle bağlı olarak Microsoft tarafından karar veriliyor.

Resim-1

Öncelikle bir paket oluşturup bunun test ortamına taşınmış olması gerekiyor. Önceki yazılarımda nasıl taşındığını anlatmıştım. Sonrasında Environment changes ile bu yüklemleri listeleyelim.

Resim-2

Burada ki paketlerden canlıya almak istediğinizi Mark as release candidate ile işaretliyoruz.

Resim-3

Sonra canlı ortamı açıp Maintain -> Update Environment’a tıklıyoruz.

Resim-4

Burada işaretlediğimiz paketi seçip hangi saatle alınmasını istiyorsak onu seçiyoruz.

Resim-5

Schedule ile tamamladıktan sonra ortamınızda aşağıdaki gibi bir sekme çıkacak. Zamanı geldiğinde işlem Microsoft tarafından yapılacak.

Resim-6

İşlem bittikten sonra Environment change history sayfasından detaylara bakabilirsiniz. Aktarım 44 dakika sürmüş. Eskiye göre bu süre çok azaldı ama hala gidilecek yol var.

Resim-7

Bu yazıda Sel-service tipindeki ortamlarda canlıya geliştirmelerin nasıl taşınacağını anlatmaya çalıştım.

Selamlar.

www.fatihdemirci.net

TAGs: Microsoft Life Cycle Services, LCS, Azure, Azure DevOps, Live deployment, Microsoft Dynamics 365, MsDyn365FO, MsDyn365CE, MsDyn365

Ax2012’den Dynamics 365 Kod Yükseltmesi Nasıl Yapılır? -1

Bu yazıda Ax2012’den Dynamics 365 Finance and Operations’a kod yükseltmesi nasıl yapılır anlatmaya çalışacağım. Ax2012 R3’den eski versiyonlardan bir yükseltme yapmak istiyorsanız öncelikle Ax2012R3’ e kodlarınız yükseltip oradan burada anlatacağım adımları uygulamanız gerekiyor. Kod yükseltme çok uzun süren ve zahmetli bir iş buna göre hazırlık yapmak lazım. Benim tavsiyem projeye başlamadan önce nelerin yeni versiyona aktarılacağına karar verip temiz bir ortamda kodu Extension mantığına uygun hale getirmek. Ben bu yükseltme örneğinde basit bir modül için uygulanması gereken adımları anlatacağım. Tek yazıda toparlamak istemedim çok uzun bir yazı olurdu birkaç bölümde gerekli adımları anlatacağım.

Öncelikle LCS de bir proje oluşturmalıyız.

Resim-1

Read more

X++ Nesne İsimlendirme Kuralları Nelerdir?

Bu yazıda Dynamics 365 Finance and Operations geliştirmelerinde kullanılan yazılım dili olan X++’ın nesne isimlendirme standartlarından bahsedeceğim. İsimlendirme bir uygulamanın kod kalitesini gösteren en temel konulardan biridir. Doğru bir isimlendirme standardı olmayan bir uygulamanın sorunlu bir yapıya dönüşmesi çok daha muhtemeldir.

Genel kurallar şöyle sıralayabilirim.

  • Tüm isimler ABD İngilizcesi olmalıdır. Bunun sebebi genel bir standart oluşması ve projeye destek verecek mühendislerin daha kolay anlayabilmesidir.
  • Mantıksal ve açıklayıcı adlar kullanılmalıdır. Nesnenin adıyla yaptığı iş uyumlu olmalıdır. Mesela günlük aktiviteleri tutan bir tablo için: DmrActivityListTable
  • İsimlerde 40 karakterlik bir sınırı vardır.
  • Uygulama Nesne Ağacındaki (AOT) ve X++ kodundaki adlar, ABD İngilizcesi kullanılarak adlandırılır ve kullanıcı ara yüzündün de gösterilen etiketlerle uyumlu olmalıdır.
  • İsimlendirmede yazım hatası olmamalıdır. Ayrıca yapılan işe uygun jargonlar kullanılmalıdır.
  • İlişkili değişken ve nesne isimleri tüm uygulama boyunca aynı olmalıdır.
  • AOT içindeki nesneler tekil isime sahip olmalıdır.
  • Kullanıcı ara yüzünde görünen tüm metinler bir etiket kullanılarak tanımlanmalıdır.
  • Yeni nesneler için bir ön ek tanımlanmalı ve tüm nesnelerde kullanılmalıdır. Genelde 2 veya 3 karakterlik bir ön ek tespit edilir. Ben firmamda Dmr kullanıyorum.
  • Tablo, EDT, Enum veya Sınıf ile aynı ada sahip olamaz.
  • CamelCase kullanılır. Örnek vermek gerekirse: Tablo: LedgerJournalTrans. Değişken: amountCurTotal. EDT: CustAcount
  • Alt çizgi (_) sadece türetilmiş nesnelerde, metoda parametre olarak gelen değişkenlerde kullanılır ve lokalizasyon eklerinde kullanılır.
  • Bir etiketin ilk harfi büyük olmalı ve diğer tüm harfler küçük olmalıdır.

Mümkün olduğunca nesne adları üç temel bileşenden hiyerarşik olarak oluşturulmalıdır:

{İş alanı adı} + {İş alanı açıklaması} + {sınıf için gerçekleştirilen eylem, tablo için içerik}

Örnekler:

  • CustInvoicePrintout
  • PriceDiscAdmCopy
  • PriceDiscAdmDelete
  • PriceDiscAdmSearch
  • PriceDiscAdmName
  • LedgerJournalTable
  • LedgerJournalTrans

Kısaltmalardan mümkün olduğunca kaçınmak gerekiyor. Eğer kısaltma kullanacaksanız tüm sistemde aynı kısaltmayı kullanmalısınız. Sistemde bulunan bazı kısaltmalar şunlardır:

  • Customer -> Cust
  • Payment -> Paym
  • Bill of material -> BOM
  • Number -> Num
  • Warehouse Management System -> WMS

Değişken isimlendirmede kısaltma veya ön ek kullanılmamalıdır. Değişkenin kullanım amacına uygun isimlendirme yapılmalıdır.

Doğru örnekler:

  • CustTable                           custTableUpdate;
  • CustInvoiceJour               custInvoiceJour;
  • CustAcount                       custAcountInvoice;
  • Amount                              totalTaxAmount;

Hata örnekler:

  • CustTable                           ct;
  • CustInvoiceJour               ciJour;
  • CustAcount                       customer;
  • Amount                              amount1;

Extension için ayrı bir makale bile yazılabilir. Hala netleşmeyen konular olmakla birlikte temel olarak şu makaleyi takip edebilirsiniz. Burada önemli olan tüm geliştiricilerin aynı standardı kullanması. Aksi durumda kod tekrarı ve gereksiz zaman kaybı oluşabilir.

Bu yazıda isimlendirme standartlarından bahsettim. Birçok kural aslında genel yazılım kuralı farklılaşanlar ürünün getirdiği standartlar, bunlar doğrudur diğer kullanımlar hatalıdır diye bir durum yok aslında. Herkesin farklı bir yoğurt yiyişi olabilir ama Dynamics 365 bir ürün ve standartları var, kendi standartlarınızı buraya uygulamamalısınız. Sonuçta burası bir uygulama ve biz ona eklemeler yapıyoruz. Dolayısıyla standartlarına uymak zorundayız. Bu konu gerçekten çok önemli doğru isimlendirme olmayan bir kodda eklem yapmak hata ayıklamak gerçekten çok can sıkıcı oluyor.

Selamlar.

www.fatihdemirci.net

TAGs: Microsoft Life Cycle Services, LCS, Azure, Azure DevOps, Nameing, Microsoft Dynamics 365, MsDyn365FO, MsDyn365CE, MsDyn365, Dynamics 365 Insights Power BI, Power Automate, Power Apss, Power Virtual Agents, Dynamics 365 nedir, Dynamics 365 ERP, Dynamics 365 CRM

Microsoft Kendi Yönettiği Tier 1 ortamları kaldırıyor!

Önceki yazılarımda bir Dynamics 365 FO satışında sonra Microsoft tarafından 3 ortam sağlandığını anlatmıştım. Bunlardan biri Tier1 tipinde ve bizim genelde Build ortamı olarak kullandığımızı ortamdı. Geçen gece yarısı gelen aşağıdaki maile bu ortamların 1 Aralık itibariyle kaldırılacağını öğrendik.

As communicated previously, Microsoft is removing the use of Remote Desktop Protocol (RDP) to access environments managed by Microsoft. As RDP access is required for development, going forward customers will be required to develop using a Cloud Hosted Environment or download a local “Virtual Hard Disk” (VHD) within Lifecycle Services. Cloud Hosted Environments will allow customers to manage the compute, size, and cost of these environments. This infrastructure change will ensure that customers decouple development tools from any running environment.

In addition, effective November 1, Tier 1 environments will not be included in the purchase of Dynamics 365 Finance, Dynamics 365 Supply Chain Management, Dynamics 365 Project Operations, or Dynamics 365 Commerce apps. The ability to purchase additional Add-On tier 1 environments will also be removed at this time. Beginning December 1, Remote Desktop Protocol (RDP) access for the existing Tier 1 Developer environments, managed by Microsoft, will be removed and decommissioned. Customers will need to preserve or move data within these environments by this date. See the FAQ below with links to existing documentation.

Microsoft will continue to invest in development tools and processes to allow customers to extend the rich capabilities available within Dynamics 365. Learn about one of these key investments, which allows for build automation that uses Microsoft-hosted agents and Azure Pipelines. This approach helps customers avoid the setup, maintenance, and cost of deploying build virtual machines (VMs). It also reuses the existing setup of build agents to run other .Net build automation.

Azure credits will be provided for qualifying customers to use for deploying Tier 1’s using Cloud Hosted Environments. Complete this survey to submit your request.

Resim-1

RDP’nin kaldırılacağı çok uzun zamandır konuşuluyordu ama direk makinelerin kaldırılıyor olması beni biraz şaşırttı üstelik kısa bir sürede. Bu arada direk 1 Aralıkta tüm makineler silinmeyecek kullanım durumlarına bakılıp zamanla kaldırılacak diyorlar ama buna güven olmaz tedbir almak lazım.

Bu durumda ne yapmak lazım derseniz aslında cevap bu makineyi nasıl kullandığınıza bağlı.

Eğer sadece Build için kullanıyorsanız iki seçenek var.

  1. Kendi Azure hesabınızda yeni bir Build makine deploy etmek. Bunu önceki yazılarımda anlatmıştım. Buradan okuyabilirsiniz.
  2. Azure-hosted builds ayarlayabilirsiniz. Tabi bunun bazı kısıtları var. Eğer Test ve DB Senkronizasyonu yapıyorsanız bu işinizi görmez.

Eğer Dev makinesi olarak kullanıyorsanız ve bulut istiyorsanız tek seçenek şu anda Burada anlattığım kendi hesabınızda yeni bir makine ayağa kaldırmak. Makineye bağlı olamayan bir geliştirme aracı üzerinde çalışmalar devam ediyor ama net bir takvim henüz yok.

Bu karar proje maliyetlerini arttıracak ben lisans boyutuna göre birden çok geliştirme ortamının ücretsiz gelmesi gerektiğini düşünüyordum elimizdekinden de olduk.

Selamlar.

www.fatihdemirci.net

TAGs: Microsoft Life Cycle Services, LCS, Azure, Azure DevOps, Microsoft Dynamics 365, MsDyn365FO, MsDyn365,

Dynamics 365 Finance and Operations Task Recorder Nedir?

Bu yazıda Dynamics 365 Finance and Operations çok önemli araçlarından biri olan Task Recorder’dan bahsedeceğim.

Task Recorder kullanıcıların birçok farklı durum için iş süreçlerini kaydetmesine olanak sağlayan bir araçtır. Aşağıdaki bazı örnek kullanımlar mevcut:

  • Uygulamanın içindeki belirli bir iş sürecinin adım adım rehber oluşturma
  • Bir iş sürecinin, isteğe bağlı olarak ekran görüntüleri içerebilen bir Microsoft Word belgesi olarak dokümantasyonu
  • Bir iş süreci için regresyon testleri
  • Uygulamada bir iş sürecinin otomatik olarak oynatılması

Finance and Operations uygulamaları için Task Recorder çok işlevsel bir araçtır. Örnekler yaparak kullanımlarını anlatmaya çalışacağım. Ayarlar ikonundan Task Recorder’ı açıyoruz.

Resim-1

Read more

Page 6 of 40« First...45678102030...Last »